R中是否存在三元运算符?

eyk*_*nal 162 r operators

正如问题所问,R中的控制序列是否类似于C的三元运算符?如果是这样,你如何使用它?谢谢!

koh*_*ske 282

作为if函数R并返回最新的评估,if-else等同于?:.

> a <- 1
> x <- if(a==1) 1 else 2
> x
[1] 1
> x <- if(a==2) 1 else 2
> x
[1] 2
Run Code Online (Sandbox Code Playgroud)

R的力量是矢量化.三元运算符的向量化是ifelse:

> a <- c(1, 2, 1)
> x <- ifelse(a==1, 1, 2)
> x
[1] 1 2 1
> x <- ifelse(a==2, 1, 2)
> x
[1] 2 1 2
Run Code Online (Sandbox Code Playgroud)

开个玩笑,你可以定义c风格?::

`?` <- function(x, y)
    eval(
      sapply(
        strsplit(
          deparse(substitute(y)), 
          ":"
      ), 
      function(e) parse(text = e)
    )[[2 - as.logical(x)]])
Run Code Online (Sandbox Code Playgroud)

在这里,你不需要关心括号:

> 1 ? 2*3 : 4
[1] 6
> 0 ? 2*3 : 4
[1] 4
> TRUE ? x*2 : 0
[1] 2
> FALSE ? x*2 : 0
[1] 0
Run Code Online (Sandbox Code Playgroud)

但是你需要括号来分配:(

> y <- 1 ? 2*3 : 4
[1] 6
> y
[1] 1
> y <- (1 ? 2*3 : 4)
> y
[1] 6
Run Code Online (Sandbox Code Playgroud)

最后,你可以用c做非常相似的方式:

`?` <- function(x, y) {
  xs <- as.list(substitute(x))
  if (xs[[1]] == as.name("<-")) x <- eval(xs[[3]])
  r <- eval(sapply(strsplit(deparse(substitute(y)), ":"), function(e) parse(text = e))[[2 - as.logical(x)]])
  if (xs[[1]] == as.name("<-")) {
    xs[[3]] <- r
        eval.parent(as.call(xs))
  } else {
    r
  }
}       
Run Code Online (Sandbox Code Playgroud)

你可以摆脱括号:

> y <- 1 ? 2*3 : 4
> y
[1] 6
> y <- 0 ? 2*3 : 4
> y
[1] 4
> 1 ? 2*3 : 4
[1] 6
> 0 ? 2*3 : 4
[1] 4
Run Code Online (Sandbox Code Playgroud)

这些不是日常使用,但可能有助于学习一些R语言的内部.


Ric*_*ton 22

像其他人一样说,使用ifelse,但你可以定义运算符,以便你几乎拥有三元运算符语法.

`%?%` <- function(x, y) list(x = x, y = y)
`%:%` <- function(xy, z) if(xy$x) xy$y else z

TRUE %?% rnorm(5) %:% month.abb
## [1]  0.05363141 -0.42434567 -0.20000319  1.31049766 -0.31761248
FALSE %?% rnorm(5) %:% month.abb
## [1] "Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec"
# or, more generally
condition %?% value1 %:% value2
Run Code Online (Sandbox Code Playgroud)

如果你定义没有%标志的运算符,它实际上是有效的,所以你可以

`?` <- function(x, y) if(x) y[[1]] else y[[2]]
`:` <- function(y, z) list(y, z)

TRUE ? rnorm(5) : month.abb
## [1]  1.4584104143  0.0007500051 -0.7629123322  0.2433415442  0.0052823403
FALSE ? rnorm(5) : month.abb
## [1] "Jan" "Feb" "Mar" "Apr" "May" "Jun" "Jul" "Aug" "Sep" "Oct" "Nov" "Dec"
Run Code Online (Sandbox Code Playgroud)

(这是有效的,因为优先级:低于?.)

不幸的是,这会破坏现有的帮助和序列运算符.


Ups*_*ide 10

if如果按以下方式使用,则与未矢量化的 ifelse 一样工作:

`if`(condition, doIfTrue, doIfFalse)
Run Code Online (Sandbox Code Playgroud)

与 ifelse 相比,使用它的优点是当向量化妨碍时(即我有标量布尔值和列表/向量结果)

ifelse(TRUE, c(1,2), c(3,4))
[1] 1
`if`(TRUE, c(1,2), c(3,4))
[1] 1 2
Run Code Online (Sandbox Code Playgroud)


Pau*_*tra 5

我会看一下ifelse命令。我会称它为更好,因为它也是矢量化的。使用汽车数据集的示例:

> cars$speed > 20
 [1] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
[13] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
[25] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
[37] FALSE FALSE FALSE FALSE FALSE FALSE FALSE  TRUE  TRUE  TRUE  TRUE  TRUE
[49]  TRUE  TRUE

> ifelse(cars$speed > 20, 'fast', 'slow')
 [1] "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow"
[11] "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow"
[21] "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow"
[31] "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow" "slow"
[41] "slow" "slow" "slow" "fast" "fast" "fast" "fast" "fast" "fast" "fast"
Run Code Online (Sandbox Code Playgroud)

  • 嗨,保罗——你的意思是用你的例子来展示一些关于“ifelse”的东西吗?;) (4认同)

Tom*_*mmy 5

就像恶作剧一样,你可以重新定义?操作员(几乎)像三元运算符一样工作(这是一个糟糕的想法):

`?` <- function(x, y) { y <-substitute(y); if(x) eval(y[[2]], parent.frame()) else eval(y[[3]], parent.frame()) }

x <- 1:3
length(x) ? (x*2) : 0
x <- numeric(0)
length(x) ? (x*2) : 0

for(i in 1:5) cat(i, (i %% 2) ? "Odd\n" : "Even\n")
Run Code Online (Sandbox Code Playgroud)

...但是你需要将表达式放在括号中,因为默认优先级与C中不一样.

只需记住在完成播放后恢复旧的帮助功能:

rm(`?`)
Run Code Online (Sandbox Code Playgroud)