QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 3800|回复: 0
打印 上一主题 下一主题

《R语言入门与实践》第十章:向量化编程

[复制链接]
字体大小: 正常 放大

1178

主题

15

听众

1万

积分

  • TA的每日心情
    开心
    2023-7-31 10:17
  • 签到天数: 198 天

    [LV.7]常住居民III

    自我介绍
    数学中国浅夏
    跳转到指定楼层
    1#
    发表于 2021-11-24 19:54 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
             《R语言入门与实践》第十章:向量化编程. I) @6 H: R6 ]: b) `8 t+ M. M
    前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行* Y0 I; B8 z7 N* |+ O6 ]5 D- W

    来达到编写高效的代码的目的.! k: w  K8 f2 W: }% i5 N7 n8 I  r
    这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.# t0 Z7 H% u4 G6 q! a1 U2 P
    通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)( J# T$ b+ Y) T* U) K. v) i6 o3 T
    功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))
      V/ P6 W$ S" p' _功能:输入一个语句,返回执行该语句所耗费的时间.

    向量化代码向量化代码的定义

    可以接受一个含有多个值的向量作为输入,并且同时操作向量中的每一个元素

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.
      3 A! y- S1 }5 ^) `- S
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    , s: E4 q+ a& R) v, Bfor(i in 1:length(vec)){0 U8 L% y( m. B  B/ T
    if(vec == "DD"){
    : v- O3 Y, e: Lvec <- "joker"
    5 v7 h: L$ c: A' ~, |; H. _}else if(vec == "C"){# `1 h6 n: x% ^6 e7 b$ U
    vec <- "ace"; j  g; q+ N: X! W! x
    }else if(vec == "7"){
    6 e% S9 {' V3 G; @7 vvec <- "king"# |8 ^+ W- I# q
    }else if(vec == "B"){" |. ?+ d1 K; X
    vec <- "queen"
    0 @: H: g$ a# }; {}else if(vec == "BB"){
    3 G4 h- W1 b! W. y; D# Fvec <- "jack"
    . U. \+ |  ^2 U! E' i  u$ X8 z" E6 M0 N3 H}else if(vec == "BBB"){
    & R% Z/ J' V( O9 e) e& F' S: @vec <- "ten"
    5 W  a+ }4 s! Y0 O8 p" w  X" x7 o}else{
    . R! J7 Y7 {( i: Wvec <- "nine"
    9 X1 Y3 E6 J5 c$ }1 V+ M/ r}8 u+ k% t: j7 d2 O2 T
    }
    ; g, W/ @5 S. Hvec
    / a  h% H: H) r) @}

    程序②——向量化

    change_vec <- function(vec){, l9 i6 Q8 I/ Q3 M( _7 i8 [+ P
    prob <- c("DD" = "joker", "C" = "ace"...)
    / [8 h2 [( y  r3 Gunname(prob[vec])) j  P, l  F' P/ `# r0 J5 U
    }

    / A# D- d6 ?& D) `% W
    方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作$ S  ~$ U& K0 X& c1 g  z) o/ z
    案例:% v; Y( k# R: p* [$ b% g, {& v' |
    程序①——未经向量化

    abs_loop <- function(vec){
    4 D0 f$ @  Q% w" l0 v% }for(i in 1:length(vec)){6 \1 W" `: T9 K
    if(vec < 0){3 `% N1 a2 H4 Y" }
    vec <- -vec
    ( O- q2 l0 X8 ]" E4 ~% V2 ]}
    3 h3 t) q3 C2 k: t; ^0 z}7 z4 h- x1 @( U& \: R/ f& H" D& f/ ]
    vec! @: T( |# @2 ^/ J2 n4 R
    }

    程序②——向量化

    abs_set <- function(vec){
    9 l5 M1 _0 p; j) q1 `/ E1 j2 U% cnegs <- vec < 0) w( j' l8 W$ [7 E5 G2 }' u0 j. F
    vec[negs] <- vec[negs] * -13 N3 |8 B3 ~7 l/ \1 X
    vec1 y) K( N2 E% H# @2 l$ |) M' L
    }

    未向量化的程序:' g8 [) l- o4 I, U: f7 T4 [6 L4 T# J0 P
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素7 ^; j# D, \8 r: L
    向量化的程序:
    + `( I/ H  n- m* S2 U# B其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

    如何在 R 中编写出快速的 for 循环原则:
    • 能放在循环外的代码,就一定不要放在循环内
    • 确保用来储存循环输出结果的对象必须具备足够的容量,以容纳循环的结果, ?- i2 y4 T- c: ~. `6 P
    范例:一个循环 1000000 次并赋值的 for 循环

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.
    1 R/ I) J& B9 i* v' D/ p在 for 循环之中, 对于对一个向量中的元素进行相应的操作.


    * i) P1 c% C1 ^  K
    + W7 R7 G6 ^- G
    + ~# C- g$ W; H
    zan
    转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-4-15 05:20 , Processed in 0.304536 second(s), 51 queries .

    回顶部