比较.条件满足时重置序列的有效方法(R)

lnN*_*oam 4 r data-manipulation

问题:

(1,2)如果满足条件(主题更改),我想重置序列.
我有forif循环,这将做到这一点,但不出所料,该方法非常慢.任何建议(例如,涉及申请家庭)是否有更有效的方法?

当前:

  subj odd_even
    a         
    a         
    a         
    b         
    b         
    b         
    b         
    c         
    c         
    c         
Run Code Online (Sandbox Code Playgroud)

目标:

  subj odd_even
    a      1   
    a      2   
    a      1   
    b      1   
    b      2   
    b      1   
    b      2   
    c      1   
    c      2   
    c      1   
Run Code Online (Sandbox Code Playgroud)
df = data.frame( subj = c("a","a","a","b","b","b","b", "c","c","c"), odd_even = "" )
Run Code Online (Sandbox Code Playgroud)

Rol*_*and 6

我喜欢这个sequence功能:

df$odd_even <- 2L - sequence(table(df$subj)) %% 2L
Run Code Online (Sandbox Code Playgroud)

data.table是另一种选择:

library(data.table)
setDT(df)
df[, odd_evenDT := 2L - seq_along(.I) %% 2L, by = subj]
Run Code Online (Sandbox Code Playgroud)

基准:

set.seed(42)
df <- data.frame(subj = sort(sample(as.character(1:1e4), 1e5, TRUE)))
DT <- data.table(df)

library(microbenchmark)
microbenchmark(roland1 = 2L - sequence(table(df$subj)) %% 2L,
           roland2 = DT[,2L - seq_along(.I) %% 2L, by = subj],
           roland3 = 2L - sequence(rle(as.integer(df$subj))$lengths) %% 2L,
           jeremy = df %>% group_by(subj) %>%
             mutate(odd_even = 2 - (row_number() %% 2)),
           frank = 2L - ave(as.integer(df$s),df$s,FUN=seq_along) %% 2L, 
           flick = ave(seq_along(df$subj), df$subj, FUN=function(x) rep(c(1,2), length.out=length(x))),
           times = 10, unit = "relative")

# Unit: relative
#     expr      min       lq      mean   median        uq      max neval
#  roland1 5.820459 5.754497 5.0368686 5.404110 4.0853039 4.847161    10
#  roland2 1.110919 1.057952 0.9840653 1.037428 0.7939004 1.176258    10
#  roland3 1.000000 1.000000 1.0000000 1.000000 1.0000000 1.000000    10
#   jeremy 5.024087 4.941366 4.3491117 4.635534 3.5144515 4.277011    10
#    frank 2.036816 1.944603 1.7809168 1.831937 1.6459597 1.607283    10
#    flick 3.655127 3.621457 3.2453089 3.473188 2.7717947 3.198285    10
Run Code Online (Sandbox Code Playgroud)


Fra*_*ank 5

这是另一个笨重的方法:

df$odd_even <- 2L - ave(as.integer(df$s),df$s,FUN=seq_along) %% 2L
Run Code Online (Sandbox Code Playgroud)

ave使每个组内的计数器.那个反击是我们奇怪的甚至是偶数测试.