QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

    自我介绍
    数学中国浅夏
    跳转到指定楼层
    1#
    发表于 2021-11-24 19:54 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
             《R语言入门与实践》第十章:向量化编程: z9 Q8 e, _3 |) O, f4 B- \
    前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行: W+ R/ B/ H' c- _' u

    来达到编写高效的代码的目的.& L1 Q. P! ]) D
    这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.
    7 x' J# ^' \$ {6 [通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)
    7 B  [9 c& A# h+ M% T功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))8 Y. g. o2 d& m
    功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.
      + k* k& t- E  \7 r9 t: V
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){
    7 T$ r7 q  U5 h2 K5 u( G" i8 jfor(i in 1:length(vec)){2 I5 E$ B! L5 m  F% x
    if(vec == "DD"){
    7 ]: e& j* r& x" Hvec <- "joker"
    & q% o9 w' ^! b& R, Y7 J5 z}else if(vec == "C"){% ?0 Z! C" E- q
    vec <- "ace"
    / B% V3 e" x+ n4 o& j9 w1 V}else if(vec == "7"){( `% w- v4 i- O
    vec <- "king"8 E, ^1 P. Y1 |# D  p" \
    }else if(vec == "B"){
    % R2 ]# {/ O: F7 b1 W3 Mvec <- "queen"9 O" L6 {* i- d7 B2 O- V
    }else if(vec == "BB"){
    ' m5 `& v) ?# J# b$ |6 G2 i6 Fvec <- "jack"
    ! a9 d! g% \( ?) v  s}else if(vec == "BBB"){3 ~1 p- S3 l6 \1 A; B
    vec <- "ten"8 C2 Q- i' d7 b, \- p! B; `  J8 r) }
    }else{
    8 k" ]# d& v! G4 uvec <- "nine"& ?! K. u* g, Y9 |7 E* I0 O1 a7 u
    }
    6 P% a9 I0 q# O& E  |% N}+ j" h, c- u4 e6 V4 L, N
    vec
    # F, M- e8 c! j9 J7 U}

    程序②——向量化

    change_vec <- function(vec){
    ' z% o) H$ O+ u! d  `+ |prob <- c("DD" = "joker", "C" = "ace"...)
    & b8 F' k& \; F  U# ?: l7 X$ hunname(prob[vec])
    0 Z8 E; }+ X0 r# M: U}

    - k9 I* L" x" m& ]( e/ c
    方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作+ S+ h" _1 T4 V. f: M- y
    案例:& q' V% l. j" w7 \
    程序①——未经向量化

    abs_loop <- function(vec){$ q# F* K8 g( }; W
    for(i in 1:length(vec)){
    ' D! V1 y% \* [4 gif(vec < 0){3 t# l; N8 C6 x5 @) B5 z) @
    vec <- -vec
    : n. \5 b% M- m}' M( b1 \  J/ a; V4 v& K1 e
    }* g( B$ N. k) U" T( p
    vec+ F3 ^6 h1 b, O$ y: ^; p
    }

    程序②——向量化

    abs_set <- function(vec){3 H' w" H- @! Z8 F" I* w
    negs <- vec < 0
    - v1 |) _; w. z$ ?1 ^, [vec[negs] <- vec[negs] * -16 K' x5 i: x/ X& X# T7 R) X
    vec2 b, d7 q- O2 L$ Z& y/ B6 n
    }

    未向量化的程序:
    ! D: k; ~7 L$ r$ Tif 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素
    1 D5 Q( i, O9 A5 }/ p" {向量化的程序:3 v; `9 `8 }$ w2 e9 H# K8 Y' F. _
    其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

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

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.
    ; m6 _7 Z8 i! \% S3 v# n& J在 for 循环之中, 对于对一个向量中的元素进行相应的操作.


    1 Q" P8 e2 o: P+ R2 p* C$ _( d% f$ {4 Y: E. ?: Z
    & b% w. w, N0 v8 B
    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-10-3 12:45 , Processed in 0.437070 second(s), 50 queries .

    回顶部