R Tip: use isTRUE()
Win-Vector Blog 2018-08-02
R Tip: use isTRUE()
.
A lot of R functions are type unstable, which means they return different types or classes depending on details of their values.
For example consider all.equal()
, it returns the logical value TRUE
when the items being compared are equal:
all.equal(1:3, c(1, 2, 3)) # [1] TRUE
However, when the items being compared are not equal all.equal()
instead returns a message:
all.equal(1:3, c(1, 2.5, 3)) # [1] "Mean relative difference: 0.25"
This can be inconvenient in using functions similar to all.equal()
as tests in if()
-statements and other program control structures.
The saving functions is isTRUE()
. isTRUE()
returns TRUE
if its argument value is equivalent to TRUE
, and returns FALSE
otherwise. isTRUE()
makes R
programming much easier.
Some examples of isTRUE()
are given below:
isTRUE(TRUE) # [1] TRUE isTRUE(FALSE) [1] FALSE isTRUE(NULL) # [1] FALSE isTRUE(NA) # [1] FALSE isTRUE(all.equal(1:3, c(1, 2.5, 3))) # [1] FALSE isTRUE(all.equal(1:3, c(1, 2, 3))) # [1] TRUE lst <- list(x = 5) isTRUE(lst$y == 7) # [1] FALSE lst$y == 7 logical(0) isTRUE(logical(0)) # [1] FALSE
Using isTRUE()
one can write safe and legible code such as the following:
# Pretend this assignment was performed by somebody else. lst <- list(x = 5) # Our own sanitization code. if(!isTRUE(lst$y > 3)) { lst$y = lst$x } print(lst) # $x # [1] 5 # # $y # [1] 5
R
now also has isFALSE()
, but by design it does not mean the same thing as !isTRUE()
. The ideas is: for a value v
at most of one of isTRUE()
or isFALSE()
is set, and both are non-NA unnamed scalar logical values. (example: isTRUE(5)
, isFALSE(5)
).
Or as help(isTRUE)
puts it:
… if(isTRUE(cond)) may be preferable to if(cond) …
Note: prior to R
3.5 isTRUE
(the current version!) was defined as “isTRUE <- function(x) identical(x, TRUE)
” (please see change-log here). This seemed clever, but failed on named logical values (violating a principle of least surprise):
oldTRUE <- function(x) identical(x, TRUE) v <- TRUE oldTRUE(v) # [1] TRUE isTRUE(v) # [1] TRUE names(v) <- "condition" oldTRUE(v) # [1] FALSE isTRUE(v) # [1] TRUE
This caused a lot of problems (example taken from R3.5.0 NEWS):
x <- rlnorm(99) isTRUE(median(x) == quantile(x)["50%"]) # [1] TRUE oldTRUE(median(x) == quantile(x)["50%"]) # [1] FALSE