Подтвердить что ты не робот

Преобразование римских цифр в числа в R

В R есть большая функция as.roman в самой базовой настройке:

as.roman(79)
# [1] LXXIX

Есть ли обратная функция, которая преобразует римские цифры в числа?

(я знаю, что могу написать это сам, но я предпочитаю использовать уже подготовленные или, желательно, стандартные функции, к сожалению, их не найти).

4b9b3361

Ответ 1

as.roman() возвращает объект класса roman, поэтому R распознает его как таковой. Вы можете сразу повернуть его обратно на арабскую цифру с помощью as.numeric(). Если у вас есть строка, которая соответствует критериям, так что она может быть действительной римской цифрой, вы можете принудительно ее применить к классу римского объекта с помощью as.roman(), а затем принудительно ввести его в арабскую цифру, создав функции принуждения. Рассмотрим:

> as.roman(79)
[1] LXXIX
> x <- as.roman(79)
> x
[1] LXXIX
> str(x)
Class 'roman'  int 79
> as.roman("LXXIX")
[1] LXXIX
> as.numeric(as.roman("LXXIX"))
[1] 79

Ответ 2

Из as.roman кода вы можете найти .roman2numeric, и его код можно увидеть, если вы запустите getAnywhere(".roman2numeric")

Код:

function (x) 
{
  romans <- c("M", "CM", "D", "CD", "C", "XC", "L", "XL", "X", 
              "IX", "V", "IV", "I")
  numbers <- c(1000L, 900L, 500L, 400L, 100L, 90L, 50L, 40L, 
               10L, 9L, 5L, 4L, 1L)
  out <- integer(length(x))
  ind <- is.na(x)
  out[ind] <- NA
  if (any(!ind)) {
    y <- toupper(x[!ind])
    y <- gsub("CM", "DCCCC", y)
    y <- gsub("CD", "CCCC", y)
    y <- gsub("XC", "LXXXX", y)
    y <- gsub("XL", "XXXX", y)
    y <- gsub("IX", "VIIII", y)
    y <- gsub("IV", "IIII", y)
    ok <- grepl("^M{,3}D?C{,4}L?X{,4}V?I{,4}$", y)
    if (any(!ok)) {
      warning(sprintf(ngettext(sum(!ok), "invalid roman numeral: %s", 
                               "invalid roman numerals: %s"), paste(x[!ind][!ok], 
                                                                    collapse = " ")), domain = NA)
      out[!ind][!ok] <- NA
    }
    if (any(ok)) 
      out[!ind][ok] <- sapply(strsplit(y[ok], ""), function(z) as.integer(sum(numbers[match(z, 
                                                                                            romans)])))
  }
  out
}

Вы можете получить доступ к .roman2numeric и преобразовать римское число в десятичные числа, как это предложил @rawr в его комментарии.

> utils:::.roman2numeric("III")
[1] 3
> utils:::.roman2numeric("XII")
[1] 12
> utils:::.roman2numeric("LXXIX")
[1] 79

Ответ 3

Номера roman в R, согласно документы:

объекты класса "roman", которые внутренне представлены как целые числа, и имеют подходящие методы для печати, форматирования, подмножества и принуждения до character.

Таким образом, вы можете получить целочисленное значение с помощью as.integer():

as.integer(as.roman(79)+as.roman(12))