QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

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

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行8 _1 b3 f) K( J: y- d5 m

    来达到编写高效的代码的目的.2 z4 P# }& H# n' l9 I$ l" _$ \+ O
    这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素., w; x$ S  v% z8 t
    通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)9 @' S3 u, A1 R, S% g
    功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))2 P/ x4 R4 q$ q0 B& y
    功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.
      ! t% j$ I2 T# N' q# z8 K9 C- v
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    8 b$ s" F; q$ \& [3 k" tfor(i in 1:length(vec)){
    / |$ N: r2 F  y! Q! Sif(vec == "DD"){
    + x: V1 q  M' [0 Z9 Uvec <- "joker"$ F( V% Z0 N8 D) g. O: x9 m
    }else if(vec == "C"){
    + X% ?1 Z: [) xvec <- "ace"
    2 L- @- T: E: |}else if(vec == "7"){, F* o9 l' d# F# R: e
    vec <- "king"
    . _7 D; N+ ?8 _- m. A}else if(vec == "B"){
    ( i7 u1 o$ Q, R1 _; E7 @8 uvec <- "queen"
    % |, \  V; l; k, L& s}else if(vec == "BB"){2 y! q- a% z; e; J' u. C4 Q
    vec <- "jack"/ e* k* S/ j( S* Z3 y( n
    }else if(vec == "BBB"){
    " b0 I' R% G$ Yvec <- "ten". q$ d4 V& ?, c! a" e' d# R* H
    }else{1 z% Y8 _# i7 p
    vec <- "nine"
    4 V  `: V  M4 E# w7 S" k: j}
    & K* A$ ]9 Y7 ]$ U}0 a9 T5 h/ F! Z4 P" m! r: w7 o8 A
    vec
    - C- P( S. ]+ X8 _/ h}

    程序②——向量化

    change_vec <- function(vec){7 \' B+ N+ s1 Z( c
    prob <- c("DD" = "joker", "C" = "ace"...)8 E/ A- }2 P; b0 a. p- A3 m- K
    unname(prob[vec])# O* r& o: W! U- \$ w& c
    }


    : f4 J+ j2 p  o. }! u方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作
    6 L/ Q, x) ^$ V/ v6 f2 V. E4 z案例:8 v1 g( ?- K6 o" i) p
    程序①——未经向量化

    abs_loop <- function(vec){2 S& Q9 J: Q& p* [' n* [
    for(i in 1:length(vec)){. P' n1 b9 Q0 O+ {
    if(vec < 0){
      r$ m0 P  s; f' pvec <- -vec
    + X" {) C, e: P4 t}/ ^  e. [% ]( s2 }1 k
    }9 v' Y) i! u# y! U1 Y0 C. |5 B- z
    vec. R1 H, I; L* {3 z' t5 L4 w  v0 f3 }3 P
    }

    程序②——向量化

    abs_set <- function(vec){
    0 J2 m& Z6 c4 E% T6 ^# J1 M% j5 ]negs <- vec < 0" o: s! W) P% }
    vec[negs] <- vec[negs] * -1$ {* e9 x4 Y0 k: j5 o
    vec: e  t) B' o5 d* h
    }

    未向量化的程序:7 h7 n6 K0 H( f: ^/ X
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素
    + d6 U3 z2 o1 r$ `! i8 k& ^向量化的程序:
    2 O9 S- G4 w! E) k7 ]" ^8 ^其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

    如何在 R 中编写出快速的 for 循环原则:
    • 能放在循环外的代码,就一定不要放在循环内
    • 确保用来储存循环输出结果的对象必须具备足够的容量,以容纳循环的结果
      3 i3 Q+ Y' Z) O+ a; t
    范例:一个循环 1000000 次并赋值的 for 循环

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.
    8 {$ c! n; z" O! x8 y在 for 循环之中, 对于对一个向量中的元素进行相应的操作.


    9 F' p3 c$ b" o( [: V
    ) l. T- Y2 R/ A# V% c/ s8 _) O' r; q- d1 W: p
    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-13 03:53 , Processed in 0.425376 second(s), 51 queries .

    回顶部