QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

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

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行9 t( ?+ W6 ?- Y3 a, K! A

    来达到编写高效的代码的目的.5 p0 `% Z4 }% {3 S7 Y
    这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.
    % G( U1 e  ]0 i: D% d& D' j通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)
    0 q5 G" y) [# L' X% m% }功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))/ j! b" g; _- @+ r- L2 b
    功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.: \9 I" b; @- v9 r: p3 I7 v( }
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){5 l$ {2 y! J+ y* q; K
    for(i in 1:length(vec)){" [" R* a: U$ N1 P
    if(vec == "DD"){
    : f; e4 U7 E; a9 ~vec <- "joker"$ g% d3 ^: d# s
    }else if(vec == "C"){" x' u2 x" K7 d8 Y
    vec <- "ace"
    ; H" D8 S6 D8 B}else if(vec == "7"){
    9 @, G1 L3 E6 P' q  zvec <- "king"* f3 K! f- j7 o6 N7 s& Q
    }else if(vec == "B"){
      P/ d1 ^. z; h2 O, X; d" _5 G$ A. n: V( yvec <- "queen"1 }( Q: F  k( l9 u9 k3 g
    }else if(vec == "BB"){5 F7 G) k) b/ O: T
    vec <- "jack"
    ' i5 h( i% o' Y# M8 F7 n2 p9 C  t$ }}else if(vec == "BBB"){2 J. n( b2 ?  y8 A  t- Z7 y1 s
    vec <- "ten"$ a$ H3 N3 @# n0 B5 \. p9 g
    }else{
    ' o0 C4 ]6 z1 b  Dvec <- "nine"
    $ m  v5 v# D. Y+ f2 `: ~, v}
      M! L9 Y6 {3 t- u}* P* M1 k- K. Z0 Z" h2 ^8 S4 E
    vec
    ( h! I+ X& C' q1 E: R}

    程序②——向量化

    change_vec <- function(vec){& U5 ?, U7 z/ |' \) h6 {  p+ e8 e- a# U
    prob <- c("DD" = "joker", "C" = "ace"...)
    : c/ n9 @3 k6 R( w2 ~unname(prob[vec])
    ' ?& K) q# }+ G2 t% D}

    : r& @9 u6 ~( f# J- k$ w+ [3 e
    方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作
    1 V) P2 ^" M& l4 S7 v案例:, m3 a1 P% f1 k- M, w/ f) \: o
    程序①——未经向量化

    abs_loop <- function(vec){# }, s) G4 T% u. [; V
    for(i in 1:length(vec)){! ]/ u; p& X6 X% b& @: m
    if(vec < 0){
    - L! _: h5 Q3 v  r9 L/ Z: r% Zvec <- -vec2 E$ n; T) t1 r, n' W
    }4 K$ M/ z5 B4 B, O
    }* r3 r" `" q- h: \$ O; A
    vec3 L/ s/ C  i0 s0 k* A- ~
    }

    程序②——向量化

    abs_set <- function(vec){
    * Y  U5 ]( |! T4 {negs <- vec < 0
    2 O: j1 M* _+ Y6 pvec[negs] <- vec[negs] * -1
      i: W# \& J" K6 L+ u7 Hvec
    7 N8 r# _- ]& s" [2 }}

    未向量化的程序:/ K/ j6 Z2 t' J6 t6 D/ u/ K8 T
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素
    3 P3 y$ S( V$ V向量化的程序:$ m* l# R4 p* j8 q5 i5 h3 t- Z
    其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

    如何在 R 中编写出快速的 for 循环原则:
    • 能放在循环外的代码,就一定不要放在循环内
    • 确保用来储存循环输出结果的对象必须具备足够的容量,以容纳循环的结果
      " ?* J6 E/ E8 s$ O
    范例:一个循环 1000000 次并赋值的 for 循环

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.& s/ G$ a, P, c/ a3 g) N1 G$ `: Q
    在 for 循环之中, 对于对一个向量中的元素进行相应的操作.

    ; L/ H. n( Y" z

    " ]& M1 B' k, T' k+ t
    , y# Z3 ^. `2 B. ]6 Y7 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, 2025-7-12 20:26 , Processed in 0.443167 second(s), 51 queries .

    回顶部