QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

    自我介绍
    数学中国浅夏
    跳转到指定楼层
    1#
    发表于 2021-11-24 19:54 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
             《R语言入门与实践》第十章:向量化编程
    8 v9 v) ~7 S9 w5 R* a前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行! \9 C! v6 h4 n3 Y

    来达到编写高效的代码的目的.5 K/ X+ ~' B" r! [; r, B! o
    这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.+ |5 g0 q# M. w$ i* ^9 F( D
    通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)7 e2 C, @% e# h5 Y
    功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))( W. [8 I- D. R; \  E: e" ~
    功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.
      + A1 y, f7 ^$ T5 c) t+ I* d
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    1 F# |, G1 w; o; c; Xfor(i in 1:length(vec)){% F2 ^! A2 _% X% V
    if(vec == "DD"){9 Y, F% o% O6 C* ^9 ^
    vec <- "joker"  ~- I1 v) o  a
    }else if(vec == "C"){
    . h0 D- h+ x( b% Svec <- "ace"
    : Y% `* ?0 \, q+ |" ?4 ]}else if(vec == "7"){; E- z. O/ Q# G, Q
    vec <- "king": Y) u+ @/ k8 \0 u0 a9 p1 S
    }else if(vec == "B"){) T' }( S) A; {7 a, w/ f
    vec <- "queen"
    2 D& W2 X; \" z/ t}else if(vec == "BB"){0 f  N  X# K7 k8 ^- g- N
    vec <- "jack") G. ^7 K3 ~, Z# p, q
    }else if(vec == "BBB"){
    * r% t/ u* p! h# Evec <- "ten"
    0 ]6 f" Y: e" }}else{
    8 J, W/ }3 [: |# Z" X7 }: ?vec <- "nine"
    9 B% {( Y( e! g$ l, A- Z3 }3 O8 j}/ z6 n1 M9 b) X
    }# }$ B5 w" Q# @& f' E
    vec
      X: ]) i) c2 F8 N6 Q, _" v& i}

    程序②——向量化

    change_vec <- function(vec){, k- U  k- y$ m* i! B9 m( X
    prob <- c("DD" = "joker", "C" = "ace"...)
    . L, e  f4 d; h- A* X9 Uunname(prob[vec])
    / U: |: e0 z7 K" x" w( ?& Y0 y}

    / J- ?1 f" L- q  Z0 m: C% v
    方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作; ~/ |  O; |& _# u  A) X6 g" A0 Y; s
    案例:5 O. Y- X5 Q7 Z- f8 e
    程序①——未经向量化

    abs_loop <- function(vec){+ C: c7 k: u" @
    for(i in 1:length(vec)){; D% i6 T7 T& t+ z/ `
    if(vec < 0){7 _9 _3 W; `( m; f
    vec <- -vec
    + |9 m+ i" R6 I% @! z/ {) Z}+ v' `" T$ [% j5 Y$ O' Z2 p
    }- e# S- {& E7 ?& U- }5 [
    vec
    ' G2 E# Q  l  }}

    程序②——向量化

    abs_set <- function(vec){
    : a3 V5 z, C  o" qnegs <- vec < 0
    1 X8 A; x( D3 R! m7 _# }3 F; Qvec[negs] <- vec[negs] * -1* B4 n: |& h: {7 Q
    vec' C" I; L4 R5 Q4 h' l% k; H
    }

    未向量化的程序:4 c. t3 p- ~2 I9 i" |, Z
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素
    : [( }  D* f4 }( X7 Q# j- v' S3 M向量化的程序:
    5 d5 ?' g9 q3 u  ]6 D其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

    如何在 R 中编写出快速的 for 循环原则:
    • 能放在循环外的代码,就一定不要放在循环内
    • 确保用来储存循环输出结果的对象必须具备足够的容量,以容纳循环的结果
      ! @# M( c: K7 }0 L' f, P
    范例:一个循环 1000000 次并赋值的 for 循环

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.4 I! Z% ^% s" J2 h) M
    在 for 循环之中, 对于对一个向量中的元素进行相应的操作.

    4 Z4 s& j+ e/ L0 G5 l, H
    % U7 Z5 V* q4 ]- b
    & J# y) }/ y% \; d6 G
    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-6-4 10:05 , Processed in 0.475674 second(s), 50 queries .

    回顶部