QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 3340|回复: 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 J+ j' z/ o: @, X7 i" S3 t前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行6 A$ d0 m3 u+ D7 G6 d) ?0 q7 ^

    来达到编写高效的代码的目的.
    ; |7 W9 V+ h" V, j" n这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.9 e+ {8 p. a% }( S' E
    通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000): b$ y$ B( d, Z3 z6 N: \# ]
    功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))
    6 X4 c, @* @3 @( v功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.
      9 A& n1 A; }1 `2 k/ x- W& U
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    + y2 Y# k( T' v8 T, D! ^for(i in 1:length(vec)){$ ]2 O6 G( h, @) N- I! G0 o6 p
    if(vec == "DD"){
    0 a$ b% w) D& Lvec <- "joker": q0 r6 a3 l* y
    }else if(vec == "C"){
    # A8 N8 n1 E& V. kvec <- "ace"
    5 v- t0 u$ {* V$ ?7 i}else if(vec == "7"){% u! d2 D  u9 n/ i
    vec <- "king"
    7 {+ B/ Q' \3 S5 \0 j: Y}else if(vec == "B"){
    7 k! m; I# Q* h3 J8 A7 c8 u. X& Vvec <- "queen"6 [$ E! d& B2 r' w* G- C2 W0 s
    }else if(vec == "BB"){
    2 {( {4 E1 \$ O! b  Qvec <- "jack"4 V/ |9 l% n& j- x" M
    }else if(vec == "BBB"){8 r( d3 @3 e  i. I) M, X$ s$ R4 q
    vec <- "ten", h% P% ?% Q) f' s/ f0 f3 e+ }
    }else{
    % E. q# A7 Y, N- ovec <- "nine"! O$ J" C9 O2 M4 \) Z
    }
    # ]& v: o' ]) W3 Q}
    - Q% l8 ?; p8 r$ h$ Lvec
    0 n0 P) v9 ?/ G! Q: Q6 t}

    程序②——向量化

    change_vec <- function(vec){
    ) N' o+ }1 S, [prob <- c("DD" = "joker", "C" = "ace"...)
    ; q2 P2 @+ C% i  f# Ounname(prob[vec])% O. X  R, k6 k; }# h; h6 U
    }


    5 S* l% G* s! |- p0 `3 A方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作
    + S$ j3 ?, o5 M  [5 k案例:3 z( m8 C$ e6 A( W$ g
    程序①——未经向量化

    abs_loop <- function(vec){! y2 U4 S. `( L9 x9 t# t$ L( [
    for(i in 1:length(vec)){1 u. [. N& F2 [5 c( @" {4 ]3 `( ~
    if(vec < 0){, X, M/ q1 V" Z" w4 v% p
    vec <- -vec( D4 Y* Q8 i3 I% m' {+ Y" W. @3 H# G5 y
    }* X* ]1 j) V4 B" z1 P
    }' i( E( A7 V3 Y
    vec6 _) |8 P9 j, @5 V
    }

    程序②——向量化

    abs_set <- function(vec){( D& e* y- ]$ ?+ H$ Y
    negs <- vec < 0
    $ m( a, R. o/ O8 z4 Mvec[negs] <- vec[negs] * -1
    8 G2 L* K% ]  S8 X& @) Yvec1 u% M8 U' y* X  T/ @
    }

    未向量化的程序:# `% T4 ?* H! H& J8 R- t1 R& d
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素
    - W, i. u$ e! j" t& W5 A9 g向量化的程序:* c- o$ f$ T. X/ ~* _2 c
    其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

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

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.3 o$ c' O2 x3 X: J  w3 P4 q
    在 for 循环之中, 对于对一个向量中的元素进行相应的操作.


      Z$ V7 A) t: n1 o" ^$ T$ K1 F1 g' S$ f3 e4 h; _

    5 H# L9 C4 D; n- j& D- I6 e% v
    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-20 22:21 , Processed in 0.600765 second(s), 50 queries .

    回顶部