QQ登录

只需要一步,快速开始

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

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

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

1158

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

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

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行$ ?$ O$ n& |$ g9 b$ M

    来达到编写高效的代码的目的.1 {6 |. B- U3 u& M8 ~8 F& \& B
    这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素., O, f, g2 s0 R' M' h; r
    通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)* b/ i7 ^* B0 }. H: S2 }% ]
    功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))$ n, A% p: C) R* c) U
    功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.1 M6 O; I1 u2 D9 W0 {
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    ; @5 R# k- T7 I6 Q! u- }for(i in 1:length(vec)){. Q+ s$ ~3 g, M; S  f/ }, h9 n$ _
    if(vec == "DD"){5 U, W. ?! l) p0 A3 Y
    vec <- "joker"
    ! K- S# i8 O! {2 T0 A}else if(vec == "C"){9 H: Y% Y2 `" {5 O: o& K1 I
    vec <- "ace"
    ( `  Z' _5 S% v7 h$ g! d& t}else if(vec == "7"){
    ) {8 ~" S5 ]' ^( F% _2 N2 Avec <- "king"* M6 `/ D0 z7 F
    }else if(vec == "B"){
    ) c) p- \7 ^) m! cvec <- "queen"4 g' ]4 U: x: U1 r
    }else if(vec == "BB"){8 H+ x8 c. j' {! Y4 ?9 n: K+ U5 y
    vec <- "jack"
    & j' w6 |. {3 \7 W}else if(vec == "BBB"){
    : q8 p$ S" W8 F; i- p# q/ k0 ovec <- "ten"
    0 d- _7 V. t. k6 p$ S}else{
    ' z/ P0 G  o/ U" y9 Jvec <- "nine"8 [7 {8 R" g) J0 a& g
    }6 s1 c' z: {" i* U- S, u' p
    }
    5 {9 w3 E) q7 G$ }$ {8 p4 {( ^. qvec& u$ C5 W+ m; Z4 j# i" G0 ?
    }

    程序②——向量化

    change_vec <- function(vec){# g& ?0 n& p' f" X0 k) N) P; @
    prob <- c("DD" = "joker", "C" = "ace"...)5 e- L5 ]: W2 A, y0 [# A+ M  l
    unname(prob[vec])0 n6 Y9 e( h9 ?
    }

    6 F0 Z, y; {: n$ Z- _5 }
    方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作3 c1 C2 @. U$ m/ c( V& T9 x( ~
    案例:
    - Q+ k6 d5 [9 q( v- i  a$ u! A程序①——未经向量化

    abs_loop <- function(vec){
    5 N4 h7 Q# K7 w* X% w# \+ @for(i in 1:length(vec)){+ E. q% ~, G8 }: Q
    if(vec < 0){
    * @$ H0 y5 N8 e, k+ s) uvec <- -vec
    * T8 \- ~# ~2 J! [' ~$ j- K9 ^; @}
    + [6 d' z) \. [}; {# G  i& s7 ]3 z' n: F
    vec5 Z2 z* d0 J- g! s9 B. K4 @
    }

    程序②——向量化

    abs_set <- function(vec){
    : q1 r* C# Y  ^6 z) F! d1 Wnegs <- vec < 0
    ) I; d9 Z7 p; `6 n& X+ i2 x) r% mvec[negs] <- vec[negs] * -12 I) }; z! ^1 ?+ _2 r
    vec6 \, J& {; K; t$ T3 J1 j
    }

    未向量化的程序:
    " a( Z* N& h' g/ yif 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素
    ( \  R  O# E: V向量化的程序:
    6 b3 s/ y1 h7 i) L* C其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

    如何在 R 中编写出快速的 for 循环原则:
    • 能放在循环外的代码,就一定不要放在循环内
    • 确保用来储存循环输出结果的对象必须具备足够的容量,以容纳循环的结果
      & m* K$ `( _+ G! i0 S
    范例:一个循环 1000000 次并赋值的 for 循环

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.3 M7 B2 {9 _( a+ K( L5 V9 F) a+ I
    在 for 循环之中, 对于对一个向量中的元素进行相应的操作.


    & r2 ?0 Z( Y* i6 o2 n- F2 w0 j% j) ^. Z" J& `% ^1 I
    7 d, _9 z- v* q4 b) D0 U
    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, 2024-5-8 18:41 , Processed in 0.303633 second(s), 50 queries .

    回顶部