罗马数字包含以下七种字符: I, V, X, L,C,D 和 M。
字符 数值
I 1
V 5
X 10
L 50
C 100
D 500
M 1000
例如, 罗马数字 2 写做 II ,即为两个并列的 1。12 写做 XII ,即为 X + II 。 27 写做 XXVII, 即为 XX + V + II 。
通常情况下,罗马数字中小的数字在大的数字的右边。但也存在特例,例如 4 不写做 IIII,而是 IV。数字 1 在数字 5 的左边,所表示的数等于大数 5 减小数 1 得到的数值 4 。同样地,数字 9 表示为 IX。这个特殊的规则只适用于以下六种情况:
给你一个整数,将其转为罗马数字。
示例1:
输入: num = 3
输出: "III"
输入: num = 4
输出: "IV"
输入: num = 9
输出: "IX"
输入: num = 58
输出: "LVIII"
解释: L = 50, V = 5, III = 3.
输入: num = 1994
输出: "MCMXCIV"
解释: M = 1000, CM = 900, XC = 90, IV = 4.
public class Solution {
readonly Tuple<int, string>[] valueSymbols = {
new Tuple<int, string>(1000, "M"),
new Tuple<int, string>(900, "CM"),
new Tuple<int, string>(500, "D"),
new Tuple<int, string>(400, "CD"),
new Tuple<int, string>(100, "C"),
new Tuple<int, string>(90, "XC"),
new Tuple<int, string>(50, "L"),
new Tuple<int, string>(40, "XL"),
new Tuple<int, string>(10, "X"),
new Tuple<int, string>(9, "IX"),
new Tuple<int, string>(5, "V"),
new Tuple<int, string>(4, "IV"),
new Tuple<int, string>(1, "I")
};
public string IntToRoman(int num) {
StringBuilder roman = new StringBuilder();
foreach (Tuple<int, string> tuple in valueSymbols) {
int value = tuple.Item1;
string symbol = tuple.Item2;
while (num >= value) {
num -= value;
roman.Append(symbol);
}
if (num == 0) {
break;
}
}
return roman.ToString();
}
}
时间复杂度:O(1)
空间复杂度:O(1)
回顾前言中列出的这 13 个符号,可以发现:
这恰好把这 13 个符号分为四组,且组与组之间没有公共的符号。因此,整数 num 的十进制表示中的每一个数字都是可以单独处理的。
进一步地,我们可以计算出每个数字在每个位上的表示形式,整理成一张硬编码表。如下图所示,其中 0 对应的是空字符串。
利用模运算和除法运算,我们可以得到 num 每个位上的数字:
thousands_digit = num / 1000
hundreds_digit = (num % 1000) / 100
tens_digit = (num % 100) / 10
ones_digit = num % 10
public class Solution {
readonly string[] thousands = {"", "M", "MM", "MMM"};
readonly string[] hundreds = {"", "C", "CC", "CCC", "CD", "D", "DC", "DCC", "DCCC", "CM"};
readonly string[] tens = {"", "X", "XX", "XXX", "XL", "L", "LX", "LXX", "LXXX", "XC"};
readonly string[] ones = {"", "I", "II", "III", "IV", "V", "VI", "VII", "VIII", "IX"};
public string IntToRoman(int num) {
StringBuilder roman = new StringBuilder();
roman.Append(thousands[num / 1000]);
roman.Append(hundreds[num % 1000 / 100]);
roman.Append(tens[num % 100 / 10]);
roman.Append(ones[num % 10]);
return roman.ToString();
}
}