QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

    自我介绍
    数学中国浅夏
    跳转到指定楼层
    1#
    发表于 2021-11-24 19:54 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
             《R语言入门与实践》第十章:向量化编程
    & A9 x# [# O3 W前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行. z6 A- x2 D% M/ `/ T/ R% R

    来达到编写高效的代码的目的.! p* A2 A9 j2 l7 K7 p% G; M! T" F
    这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.1 i3 b6 p3 V. P, c* b
    通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)
    ; _! K$ n1 K2 _4 P功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))
    $ I6 |- `" W# e. \. `+ a功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.$ N  y5 O' Q/ u, g. d
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    ) g: P$ U9 b4 D) u7 N( ?$ a- Xfor(i in 1:length(vec)){" Q+ H/ B2 \! j" {, ^) u5 L
    if(vec == "DD"){& {% T1 p) _+ p$ }0 o
    vec <- "joker"( a; P. z/ M: i9 @
    }else if(vec == "C"){* @- o* R  K+ u$ w! j
    vec <- "ace"
    ' R* @4 i5 h) S7 J}else if(vec == "7"){
    6 S( Y2 ^2 Y2 Cvec <- "king"
    ; D; P0 F- `0 k4 b/ J2 {; K}else if(vec == "B"){, j$ h9 O: G, n0 C. }
    vec <- "queen"
    . C0 S! i# k; X# Y6 m7 {! P}else if(vec == "BB"){5 X8 A: O5 f# ?! P; H! T4 i
    vec <- "jack"+ K) Q9 z8 n6 T8 \0 J3 F: H) A: n; p
    }else if(vec == "BBB"){
    : u9 I3 a! C3 ^. S: [vec <- "ten"
    0 z/ E1 h; A/ i  @$ s}else{
    " ?+ x& V% Z* X! m! Wvec <- "nine"# |9 k8 O6 p! \
    }
    & F6 G0 A- a, X: T3 X: o}
    ( X$ ?( `. T& C& R. [  [# jvec
    % s; M0 j' S5 D}

    程序②——向量化

    change_vec <- function(vec){
    + f! ?1 d4 h( }6 a, @prob <- c("DD" = "joker", "C" = "ace"...)5 C! a4 v3 a3 H4 o* ]
    unname(prob[vec])* e# @" }1 D/ {/ H: u. O8 W
    }

    8 U- {: J8 [0 M7 A: G1 u
    方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作. f1 N, s* g2 O- i" z* W% J  S: R
    案例:
    " G7 ?, z+ I3 t. i) s# }7 }0 x% g5 \程序①——未经向量化

    abs_loop <- function(vec){1 {5 b, X. R, R7 d/ G
    for(i in 1:length(vec)){
    1 b* ]* h) r5 L/ x6 T8 Lif(vec < 0){
    - p9 J: S% {% ]2 s; jvec <- -vec0 M0 V) J6 G; R4 R, Y  N* d4 r4 p- t4 r
    }# J' _. B9 M+ C; n, ]
    }3 g7 F' s  B% d9 \5 N
    vec
      s! c$ u) D. R+ a* M; D6 L}

    程序②——向量化

    abs_set <- function(vec){
    ; I6 K: n% Z3 t1 T  x* e& x3 ?/ Hnegs <- vec < 0
    8 m2 f: @+ Z, x/ Bvec[negs] <- vec[negs] * -1( V! L1 b, z  b: f1 X# d1 P
    vec
    . q: y, ?/ S4 p# P9 D}

    未向量化的程序:; p3 Y% Y3 i3 c* t: m& k9 @+ U
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素5 v3 x5 c8 i* l3 A* A
    向量化的程序:
    + T' w  E; n. o3 A其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

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

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.
    , K3 @6 W2 k1 G% {$ L& \- }在 for 循环之中, 对于对一个向量中的元素进行相应的操作.

    / Z% E9 k3 V0 ]( N
    , X9 ?( P5 H; n) U* Y

    ) Q, b& p! h; ~, T- s- \1 N( W
    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, 2025-7-15 05:33 , Processed in 0.382662 second(s), 50 queries .

    回顶部