QQ登录

只需要一步,快速开始

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

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

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

1178

主题

15

听众

1万

积分

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

    [LV.7]常住居民III

    自我介绍
    数学中国浅夏
    跳转到指定楼层
    1#
    发表于 2021-11-24 19:54 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
             《R语言入门与实践》第十章:向量化编程' [; a2 j1 ~! W' K+ r6 x% I
    前言

    利用 R 的三大法宝:

    • 逻辑判断
    • 取子集
    • 按元素方式执行6 C: T6 Z7 i: Q2 R  d

    来达到编写高效的代码的目的.
      `; {& O  _) x% s9 l; u9 w0 _5 p' B这样的代码的特点是可以接受整个向量作为输入,并同时处理向量中的元素.
    : x0 m" B1 `% R, c通过以下几个案例来阐述向量化编程

    预备知识rep() 函数

    格式: rep(c(-1, 1), 5000000)
    2 @0 ]  e6 ~7 q9 e功能:接受某个值/向量以及次数,返回该值/向量的重复执行次数长度的更长的向量

    system.time() 函数

    格式: system.time(function(object))
    8 P% D5 i  f6 o0 a/ w功能:输入一个语句,返回执行该语句所耗费的时间.

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

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

    如何编写向量化代码

    原则:

    • 尽量使用向量化的函数来完成任务:比如使用 R 库中的原函数
    • 对于重复的情况,使用逻辑值取子集的方法,而不是 for & if 的方法.
      % F. n# C# a' a5 F# P0 a
    方法一:使用向量化的函数/查找表

    程序①——未经向量化

    change_symbols <- function(vec){& f: z7 [7 R$ e2 g" a
    for(i in 1:length(vec)){
    * s6 s1 m5 I0 Eif(vec == "DD"){. w8 \4 Z. p5 s- ^
    vec <- "joker"" ?3 B- H+ o# j6 s* H2 G) k
    }else if(vec == "C"){
    : e- ?) N0 ^! W1 n6 Ivec <- "ace"1 M; R4 s1 d8 R
    }else if(vec == "7"){
    9 Q! M* K, X1 f- {0 G2 Evec <- "king"$ w, d/ c9 F; l) w& s$ I
    }else if(vec == "B"){4 E, _2 ?# m4 ]- h4 T
    vec <- "queen"
    9 ~7 {& T) e# y/ I. x+ }}else if(vec == "BB"){
    ; h  t9 Y8 N( f: Cvec <- "jack"" d$ g; s$ A; j- a- N7 S7 a5 t
    }else if(vec == "BBB"){: Z1 o- k; e  O* N3 S  \2 c% Q
    vec <- "ten"
    4 b* E" N' f$ c! ~0 G}else{
    - a0 _7 b) d% @8 A# G6 d# L1 zvec <- "nine"5 Y$ j" |6 e- d3 k8 ?8 q
    }8 ^' d* x* L$ M) F3 e/ p
    }
    " D$ @* ^' {! ~# fvec
      A& p9 K9 P' ]% q* z- z}

    程序②——向量化

    change_vec <- function(vec){
    & E8 l: L4 O( ]prob <- c("DD" = "joker", "C" = "ace"...)) \3 ]0 h5 \: F3 `. U
    unname(prob[vec])3 k+ ^  T: n) o6 `2 @$ w5 g7 Y
    }

    " I# T, ]# r- X% x2 l2 R
    方法二:逻辑值取子集

    目的:一次性完成对一类情况中的所有元素的操作* D( a$ D# I4 g# L
    案例:
      l2 \0 V5 r8 ~# o- M- U) E  {程序①——未经向量化

    abs_loop <- function(vec){
    - c0 w8 k, j; tfor(i in 1:length(vec)){; N  d) }' p! ^0 Q
    if(vec < 0){5 J$ G5 I9 H2 f8 F5 h
    vec <- -vec( \5 R; u, ~4 O6 T2 K
    }
    ( S' E: `* n+ V}
    / ]$ S" j9 o" q$ Tvec9 a7 ]; S! K0 L7 u' Y( C3 ~
    }

    程序②——向量化

    abs_set <- function(vec){
    5 M% P" N$ Y* [9 i) M9 C$ r$ c4 ^) Nnegs <- vec < 0
    4 A+ H2 M2 ^* S. \- g8 avec[negs] <- vec[negs] * -1
    % R2 @# E2 L1 C% lvec' F! t/ h* o$ v6 k$ b
    }

    未向量化的程序:0 E" o1 E3 `9 r
    if 语句一次只能针对一个元素进行判断,来判断出 vec 中为负数的元素) H/ z/ I! l4 o( B/ l4 ~. b
    向量化的程序:0 w$ U( @4 J9 \' F
    其中, vec < 0 为逻辑测试,返回一个包含 TRUE, FALSE 逻辑值的向量 negs, 通过逻辑值取子集的方法,得到 vec 中为负数的元素, 即 vec[negs].

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

    在 for 循环之前,定义好一个含有 1000000 个 NA 值的向量.0 Y8 N' W: J2 w: O9 d
    在 for 循环之中, 对于对一个向量中的元素进行相应的操作.

    9 }+ ]( K' O1 B; d

    * ]$ T% T6 [* D+ D0 |0 J$ u7 ~4 `7 x
    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-13 21:54 , Processed in 0.388841 second(s), 51 queries .

    回顶部