r - 使用(如果可能)函数方法修改基于正则表达式 (regex) 向量的向量

标签 r regex algorithm vectorization stringr

我有一个包含一些列的数据框,我想修改这些列,具体取决于它们是否匹配包含在带有正则表达式的向量中的某些模式

library(fuzzyjoin)
library(tidyverse)

 (df <- tribble(~a,
               "GUA-ABC",
               "REF-CDE",
               "ACC.S93",
               "ACC.ATN"))

#> # A tibble: 4 x 1
#>   a      
#>   <chr>  
#> 1 GUA-ABC
#> 2 REF-CDE
#> 3 ACC.S93
#> 4 ACC.ATN

根据我想粘贴文本的模式,例如,对于包含 GUA 的模式,在由点连接的链的末端粘贴“GUA001”,对于包含 REF 的模式,在链的末端粘贴“GUA002”同理,可以得到:

# This is the resulting data.frame I need

#> # A tibble: 4 x 1
#>   a             
#>   <chr>         
#> 1 GUA-ABC.GUA001
#> 2 REF-CDE.GUA002
#> 3 ACC.S93       
#> 4 ACC.ATN

我想到了一些办法。

方法#1

 # list of patterns to search
 patterns <- c("\\b^GUA\\b", "\\b^REF\\b")

 # Create a named list for recoding
 model_key <- list("\\b^GUA\\b" = "GUA001",
                   "\\b^REF\\b" = "GUA002")
 
 # Create a data.frame of regexs
 (k <- tibble(regex = patterns))

#> # A tibble: 2 x 1
#>   regex       
#>   <chr>       
#> 1 "\\b^GUA\\b"
#> 2 "\\b^REF\\b"

 # perform a regex_left_join to identify the pattern
 df %>% 
  regex_left_join(k, by = c(a = "regex")) %>% 
  mutate(
   
   across(regex, recode, !!!model_key),
   a = case_when(
    
    !is.na(regex) ~ str_c(a, regex, sep = "."),
    TRUE ~ a)
   
  ) %>% select(-regex)

#> # A tibble: 4 x 1
#>   a             
#>   <chr>         
#> 1 GUA-ABC.GUA001
#> 2 REF-CDE.GUA002
#> 3 ACC.S93       
#> 4 ACC.ATN

为什么这种方法不是最优的?原始数据框有数百万行,fuzzyjoin::regex_left_join 执行此操作花费的时间太长。

方法#2

 patron <- c("GUA001" = "\\b^GUA\\b", "GUA002" = "\\b^REF\\b")
 newtex <- c("GUA001", "GUA002")
 
 pegar <- function(string, pattern, text_to_paste) {
  
  if_else(condition = str_detect(string, pattern), 
          true = str_c(string, text_to_paste, sep = "."), 
          false = string)
  
 }
 
 map2_dfr(.x = patron, .y = newtex, ~ pegar(string = df$a, 
                                            pattern = .x, 
                                            text_to_paste = .y))
#> # A tibble: 4 x 2
#>   GUA001         GUA002        
#>   <chr>          <chr>         
#> 1 GUA-ABC.GUA001 GUA-ABC       
#> 2 REF-CDE        REF-CDE.GUA002
#> 3 ACC.S93        ACC.S93       
#> 4 ACC.ATN        ACC.ATN

reprex package 创建于 2021-05-20 (v2.0.0)

使用方法 # 2 我无法获得单个列。

作为旁注,使用 str_replace_all 并使用命名向量替换字符串中的某些值目前似乎不是一个好的选择。

有没有办法更优化地做到这一点?

最佳答案

利用 stringrpurrr 的一个选项可能是:

imap_dfr(model_key,
         ~ df %>%
          filter(str_detect(a, .y)) %>%
          mutate(a = str_c(a, .x, sep = "."))) %>%
 bind_rows(df %>%
            filter(str_detect(a, str_c(names(model_key), collapse = "|"), negate = TRUE)))

  a             
  <chr>         
1 GUA-ABC.GUA001
2 REF-CDE.GUA002
3 ACC.S93       
4 ACC.ATN  
 

关于r - 使用(如果可能)函数方法修改基于正则表达式 (regex) 向量的向量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67628146/

相关文章:

dplyr 中的重命名功能不起作用

r - 向量序列上的子集数据帧

javascript - Angular 2 NgFor Regex 消息错误 - [(ngModel)] 有效,但 ngModel 未定义

regex - 在 Excel 中使用正则表达式,我可以在替换匹配的字符串之前对匹配的模式执行一些算术运算吗?

java - 类勉强和简单类正则表达式的等价性

c++ - 从两个单链表中找到相同的节点。不能用hash,不能是O(n^2)复杂度

java - 是否有任何用于折线简化的开源 Java 库?

r - 基于节点权重构建图的优化算法

r - 为 R 函数创建进度条

java - 从 LinkedList 中删除特定元素....?