R - Извлечение числа, связанного с персонажем - PullRequest
1 голос
/ 17 марта 2020

Я пытаюсь извлечь количество атомов углерода, водорода и кислорода из химической формулы. Ранее я нашел некоторый код, который я пытался использовать. Проблема в том, что код работает только тогда, когда химическая формула содержит более 1 элемента.

V <- DATA # example: CH4O, H2O, C10H18O2
# V is a data.frame

C1 <- as.integer(sub("(?i).*?C:?\\s*(\\d+).*", "\\1", V))
# NA NA 10
H1 <- as.integer(sub("(?i).*?H:?\\s*(\\d+).*", "\\1", V))
# 4 2 18
O1 <- as.integer(sub("(?i).*?O:?\\s*(\\d+).*", "\\1", V))
# NA NA 2

В настоящее время я использую is.na(C1) <- 1, чтобы изменить NA на 1, а затем вручную изменить значения 0. Есть ли более эффективный код, который я могу использовать, чтобы получить правильное количество элементов в химических формулах (особенно в случаях, когда значение равно 0 или 1 и приводит к результатам NA). Дайте мне знать, если вам нужна дополнительная информация или мне следует изменить какой-либо формат.

РЕДАКТИРОВАТЬ: Желаемые значения будут состоять в том, чтобы получить все правильные значения без NA и вручную изменить значения на 0, если это возможно.

C1
# 1 0 10
H1
# 4 2 18
O1
# 1 1 2

РЕДАКТИРОВАТЬ2: Вот пример фактических данных, которые я импортирую

Meas. m/z   #   Ion Formula Score   m/z err [mDa]   err [ppm]   mSigma  rdb e¯ Conf Adduct  
84.080700   1   C5H10N  n.a.    84.080776   0.1 0.9 n.a.    2.0 even        
89.060100   1   C4H9O2  n.a.    89.059706   -0.4    -4.4    n.a.    1.0 even        
131.987800  1   C2H4N3P2    n.a.    131.987498  -0.3    -2.3    n.a.    6.0 even        
135.081100  1   C9H11O  n.a.    135.080441  -0.7    -4.9    n.a.    5.0 even        
135.117500  1   C10H15  n.a.    135.116827  -0.7    -5.0    n.a.    4.0 even        
136.061700  1   C5H6N5  n.a.    136.061772  0.1 0.5 n.a.    6.0 even        

В первоначальном вопросе я только перечислил V как происходящий из вектора формул, но что У меня фактически есть data.frame с другой информацией, и я использую V[,3] при выполнении вычислений, чтобы получить интересующий столбец.

Ответы [ 3 ]

2 голосов
/ 17 марта 2020

Вот альтернатива:

vec <- c("CH4O", "H2O", "C10H18O2", "C2H4N3P2")

molecules <- regmatches(vec, gregexpr("\\b[A-Z][a-z]*\\d*", vec))
molecules <- lapply(molecules, function(a) paste0(a, ifelse(grepl("[^0-9]$", a), "1", "")))

atomcounts <- lapply(molecules, function(mol) setNames(as.integer(gsub("\\D", "", mol)), gsub("\\d", "", mol)))

atoms <- unique(unlist(sapply(atomcounts, names)))
atoms <- sapply(atoms, function(atom) sapply(atomcounts, function(a) if (atom %in% names(a)) a[atom] else 0))
rownames(atoms) <- vec
atoms
#           C  H O N P
# CH4O      1  4 1 0 0
# H2O       0  2 1 0 0
# C10H18O2 10 18 2 0 0
# C2H4N3P2  2  4 0 3 2
1 голос
/ 17 марта 2020

Мы можем использовать base R методы с strsplit и xtabs

out <- do.call(rbind, Map(cbind,
   lapply(strsplit(gsub("(?<=[A-Z])(?![0-9])", "1", vec,
   perl = TRUE), "(?<=[A-Z])(?=[0-9])|(?<=[0-9])(?=[A-Z])", 
    perl = TRUE), function(x) data.frame(key = x[c(TRUE, FALSE)], 
       value = as.numeric(x[c(FALSE, TRUE)]))), grp = vec))

xtabs(value ~ grp + key, out)
#         key
#grp         C  H  O  N  P
#  CH4O      1  4  1  0  0
#  H2O       0  2  1  0  0
#  C10H18O2 10 18  2  0  0
#  C2H4N3P2  2  4  0  3  2

Или с tidyverse

library(stringr)
library(dplyr)
library(tidyr)
tibble(col1 = vec,
       col2 = str_replace_all(col1, "(?<=[A-Z])(?![0-9])", "1")) %>%
       separate_rows(col2, sep= "(?<=[A-Z])(?=[0-9])|(?<=[0-9])(?=[A-Z])") %>%
       group_by(col1) %>% 
       summarise(key = list(col2[c(TRUE, FALSE)]),
            val = list(col2[c(FALSE, TRUE)])) %>%
       unnest(c(key, val)) %>% 
   pivot_wider(names_from = key, values_from = val, values_fill = list(val = 0))
# A tibble: 4 x 6
#  col1     C     H     O     N     P    
#  <chr>    <chr> <chr> <chr> <chr> <chr>
#1 C10H18O2 10    18    2     0     0    
#2 C2H4N3P2 2     4     0     3     2    
#3 CH4O     1     4     1     0     0    
#4 H2O      0     2     1     0     0    

data

vec <- c("CH4O", "H2O", "C10H18O2", "C2H4N3P2")
1 голос
/ 17 марта 2020

Возможно, это не самый элегантный код, который я когда-либо писал, но для данной химической формулы это вернет счетчики каждого элемента в помеченном векторе. На промежуточном этапе counts.per.equation возвращается счетчик каждого элемента в каждом уравнении.

library(stringr)

extract <- str_extract_all(c('CH4O', 'H2O', 'C10H18O2'), '\\D\\d*')

counts.per.equation <- lapply(extract, function(x) {
  elements <- str_extract_all(x, '\\D+')
  counts <- str_extract_all(x, '\\d+', simplify = T)
  counts[counts == ''] <- 1
  counts <- as.numeric(counts)
  names(counts) <- elements
  return(counts)
})

total.counts <- Reduce(function(x, y) {
  names <- union(names(x), names(y))
  counts <- sapply(names, function(z) sum(x[z], y[z], na.rm = T))
  names(counts) <- names
  return(counts)
} , counts.per.equation)

 C  H  O 
11 24  4 
...