QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

    自我介绍
    数学中国浅夏
    跳转到指定楼层
    1#
    发表于 2021-11-24 19:54 |只看该作者 |正序浏览
    |招呼Ta 关注Ta
             《R语言入门与实践》第十章:向量化编程" q# P: d3 H  m  s# g1 s
    前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行
      4 w+ ?- x! ^2 {3 D

    来达到编写高效的代码的目的.
    5 }" y2 d6 d% X* ~这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.
    ) t% Z" q+ `4 A/ C% s通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)
    2 u$ g4 c9 y9 T) Y5 c# ?功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))
    2 _" m8 C' D2 L' z# P! `功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.# ]! T& e% A, P5 N" D3 _
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){8 H' [5 y: n, s( y, X  D
    for(i in 1:length(vec)){2 u7 e( `! k3 P
    if(vec == "DD"){; n2 F, h& J) I* u* L0 g5 m
    vec <- "joker"8 D" y; T& r0 W* C) x( p' u
    }else if(vec == "C"){) [$ y- h3 A/ H: A, `" B, t9 p
    vec <- "ace"
    # q2 G0 G, g' m+ J}else if(vec == "7"){
    + ^: g& C6 R4 F# H, [vec <- "king"( g$ [* q% ]/ z- K
    }else if(vec == "B"){5 B. s+ }6 U- d) y4 z
    vec <- "queen"
    - Z3 h- V3 y; U/ F; \}else if(vec == "BB"){
    0 I& c4 @  T' ~; S4 ovec <- "jack"' S! ^% x# W# d% u3 n# G9 `; f6 y3 k
    }else if(vec == "BBB"){
    + l3 a* B4 f6 |vec <- "ten"' z* J1 k2 D1 S
    }else{
    2 i7 _, l. e- ]) F0 {6 O( Z' u' `0 {vec <- "nine"& \: k8 j: |* n+ t
    }* B% C9 k* Y6 {8 [" ~% ?
    }
    , j/ Q; h) R- ^# N+ ]vec: g; g# V: {/ N$ L2 U6 ?: j) K
    }

    程序②——向量化

    change_vec <- function(vec){
    8 d# i+ }! x( U  B$ D% Aprob <- c("DD" = "joker", "C" = "ace"...)
    ' ?$ H2 z  u7 K& [* p3 }unname(prob[vec])
    % J8 }1 }5 W* \}


    7 S" V/ y) T) V7 J5 U# Y方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作) e% _& T& m# u% s
    案例:5 L, U/ i2 G" B2 U+ W
    程序①——未经向量化

    abs_loop <- function(vec){
    5 ^; t; K- V( s( d- |for(i in 1:length(vec)){$ M' N% ~+ c& t! I3 |
    if(vec < 0){( V: R, W8 s9 L# M) |' r
    vec <- -vec+ y7 q$ d( w1 s5 `# q2 p- h1 I# x3 G+ p
    }! U/ L: r0 X" D" {, {; h; m4 t
    }, T2 e' E% N1 }$ N0 F
    vec3 I. a6 k  X9 Y9 `/ @+ a4 T9 q
    }

    程序②——向量化

    abs_set <- function(vec){3 D7 P  n  F: e, l8 S
    negs <- vec < 0! _. q- Q! N) O* ^( S5 e
    vec[negs] <- vec[negs] * -1
    ' [# O; G$ \7 i, ovec
    % D' ]$ U$ u7 I}

    未向量化的程序:
    ! d* @5 u: t  W7 eif 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素( Q3 M4 W2 F6 a" Q  [1 G; _
    向量化的程序:! N5 @' f* M" Z" y1 \( T0 l
    其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

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

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.$ Q. z- @% `0 B! N& c; W/ @! T0 G5 |  ]
    在 for 循环之中, 对于对一个向量中的元素进行相应的操作.

    # Q* C# W5 g0 w, N( r) c8 i
    + K& m; `7 \2 G& K2 _- \( j

    % g9 Q! \6 H& G" y* 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, 2026-4-14 02:17 , Processed in 0.400758 second(s), 52 queries .

    回顶部