QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

    自我介绍
    数学中国浅夏
    跳转到指定楼层
    1#
    发表于 2021-11-24 19:54 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
             《R语言入门与实践》第十章:向量化编程9 W$ o/ K* L8 ^0 P, g& d
    前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行, R, S: @  r2 b# ~) B- i# Y1 s

    来达到编写高效的代码的目的.
    & I4 G- n% w/ j7 _这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.. ~4 ~7 M9 r. ^4 F0 G
    通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)
    % x! C- \# J3 ]0 V- ~; P功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))
    : q- x0 d' q* ~8 K) e功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.8 Y% k: ]$ I5 ?* ?! Q- a+ a
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    5 `$ u# i, k& e( k% Y4 J- c  rfor(i in 1:length(vec)){
    & r& b0 v9 h0 e6 K8 B9 l" Mif(vec == "DD"){
    ; o, K& V. i2 A; q9 Z+ g' N! N( [* a* gvec <- "joker"& \3 s* r$ Q! |0 ?' z
    }else if(vec == "C"){
    3 z6 a& i/ h+ M$ Svec <- "ace"/ Q4 H1 ~5 L4 a4 k9 L, P) V% w
    }else if(vec == "7"){
    ! j5 [9 A1 R) |vec <- "king"3 q% j4 G5 @  Z  Y5 M. Z
    }else if(vec == "B"){* ?7 [0 w4 q4 f# u" |; m
    vec <- "queen"5 E$ G4 t1 b5 `0 z
    }else if(vec == "BB"){
    2 i% n0 P; y3 S$ ?: y; j* \6 Pvec <- "jack"
    6 S9 K8 |, W$ N9 l}else if(vec == "BBB"){5 ?( M4 G. {8 \
    vec <- "ten"3 {# ]2 q6 l5 D' j8 u
    }else{9 l# K: E" Y6 [* {! T7 C7 ~
    vec <- "nine") }) |) [( m+ i9 v/ |1 t4 f* i
    }" X4 s, D' T4 N) Z3 v
    }+ ?# i6 G5 P1 b7 h( N  ~2 M
    vec
    5 E" c/ L. T4 U' k* f; \2 P& f7 z( F}

    程序②——向量化

    change_vec <- function(vec){  e" _5 X* N' b/ Y1 _* x3 q4 f$ T( @* [7 E
    prob <- c("DD" = "joker", "C" = "ace"...)& A/ y) N2 r1 W6 r
    unname(prob[vec]): I1 v- l9 B& f! p5 J
    }


    7 U3 w, o2 m& m- w. J0 x# s9 G方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作
    ( C: F; H' k6 H! V! d  s) t案例:) [4 P6 o, V# P' r5 h/ x& o2 z
    程序①——未经向量化

    abs_loop <- function(vec){; L1 L4 K( L' x) C2 H) i
    for(i in 1:length(vec)){% ^' I' Q9 N) Y' ~5 W2 D3 _$ A
    if(vec < 0){  g( e$ X- _; M
    vec <- -vec, `9 V$ ?/ {' F3 ?7 P# Z
    }
    # W/ }3 m# ?4 ?5 b6 ^}
    : E& K& U0 W, r1 U1 o: ivec# u5 R: j: |8 h' j
    }

    程序②——向量化

    abs_set <- function(vec){: R* m: Z, `. I5 h$ B% ^
    negs <- vec < 0
    $ C& i, |; u. ^) S6 ^vec[negs] <- vec[negs] * -1
    : E9 o$ I7 c, B6 g' \vec  A# F* L+ Q7 ?% n
    }

    未向量化的程序:% Z- Q# W  p0 j& V( w
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素
    & B$ A9 m1 Z* A) A# w  V9 b向量化的程序:5 }9 k0 R* Q, |* w- T8 M8 p8 W5 G
    其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

    如何在 R 中编写出快速的 for 循环原则:
    • 能放在循环外的代码,就一定不要放在循环内
    • 确保用来储存循环输出结果的对象必须具备足够的容量,以容纳循环的结果
      - i9 |2 i# D9 i
    范例:一个循环 1000000 次并赋值的 for 循环

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.4 a/ l5 R6 g: E! p
    在 for 循环之中, 对于对一个向量中的元素进行相应的操作.

    6 W$ O2 |5 g1 {
    ; U- Q; X: s/ U4 U) Z
    5 J7 x4 [5 ^8 R, Z, u! |" j9 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-16 08:58 , Processed in 0.594339 second(s), 51 queries .

    回顶部