[LeetCode] Roman to Integer 羅馬數字轉化成整數

Grandyang發表於2014-11-25

 

Roman numerals are represented by seven different symbols: IVXLCD and M.

Symbol       Value
I             1
V             5
X             10
L             50
C             100
D             500
M             1000

For example, two is written as II in Roman numeral, just two one's added together. Twelve is written as, XII, which is simply XII. The number twenty seven is written as XXVII, which is XX + V + II.

Roman numerals are usually written largest to smallest from left to right. However, the numeral for four is not IIII. Instead, the number four is written as IV. Because the one is before the five we subtract it making four. The same principle applies to the number nine, which is written as IX. There are six instances where subtraction is used:

  • I can be placed before V (5) and X(10) to make 4 and 9. 
  • X can be placed before L (50) and C (100) to make 40 and 90. 
  • C can be placed before D (500) and M (1000) to make 400 and 900.

Given a roman numeral, convert it to an integer. Input is guaranteed to be within the range from 1 to 3999.

Example 1:

Input: "III"
Output: 3

Example 2:

Input: "IV"
Output: 4

Example 3:

Input: "IX"
Output: 9

Example 4:

Input: "LVIII"
Output: 58
Explanation: L = 50, V= 5, III = 3.

Example 5:

Input: "MCMXCIV"
Output: 1994
Explanation: M = 1000, CM = 900, XC = 90 and IV = 4.

 

羅馬數轉化成數字問題,我們需要對於羅馬數字很熟悉才能完成轉換。以下截自百度百科:

羅馬數字是最早的數字表示方式,比阿拉伯數字早2000多年,起源於羅馬。
如今我們最常見的羅馬數字就是鐘錶的錶碟符號:Ⅰ,Ⅱ,Ⅲ,Ⅳ(IIII),Ⅴ,Ⅵ,Ⅶ,Ⅷ,Ⅸ,Ⅹ,Ⅺ,Ⅻ……
對應阿拉伯數字(就是現在國際通用的數字),就是1,2,3,4,5,6,7,8,9,10,11,12。(注:阿拉伯數字其實是古代印度人發明的,後來由阿拉伯人傳入歐洲,被歐洲人誤稱為阿拉伯數字。)
 
 
基本字元
I
V
X
L
C
D
M
相應的阿拉伯數字表示為
1
5
10
50
100
500
1000
 
1、相同的數字連寫,所表示的數等於這些數字相加得到的數,如:Ⅲ = 3;
2、小的數字在大的數字的右邊,所表示的數等於這些數字相加得到的數, 如:Ⅷ = 8;Ⅻ = 12;
3、小的數字,(限於Ⅰ、X 和C)在大的數字的左邊,所表示的數等於大數減小數得到的數,如:Ⅳ= 4;Ⅸ= 9;
4、正常使用時,連寫的數字重複不得超過三次。(錶盤上的四點鐘“IIII”例外)
5、在一個數的上面畫一條橫線,表示這個數擴大1000倍。
 
有幾條須注意掌握:
1、基本數字Ⅰ、X 、C 中的任何一個,自身連用構成數目,或者放在大數的右邊連用構成數目,都不能超過三個;放在大數的左邊只能用一個。
2、不能把基本數字V 、L 、D 中的任何一個作為小數放在大數的左邊採用相減的方法構成數目;放在大數的右邊採用相加的方式構成數目,只能使用一個。
3、V 和X 左邊的小數字只能用Ⅰ。
4、L 和C 左邊的小數字只能用X。
5、D 和M 左邊的小數字只能用C。
 
而這道題好就好在沒有讓我們來驗證輸入字串是不是羅馬數字,這樣省掉不少功夫。我們需要用到HashMap資料結構,來將羅馬數字的字母轉化為對應的整數值,因為輸入的一定是羅馬數字,那麼我們只要考慮兩種情況即可:
第一,如果當前數字是最後一個數字,或者之後的數字比它小的話,則加上當前數字。
第二,其他情況則減去這個數字。

 

解法一:
class Solution {
public:
    int romanToInt(string s) {
        int res = 0;
        unordered_map<char, int> m{{'I', 1}, {'V', 5}, {'X', 10}, {'L', 50}, {'C', 100}, {'D', 500}, {'M', 1000}};
        for (int i = 0; i < s.size(); ++i) {
            int val = m[s[i]];
            if (i == s.size() - 1 || m[s[i+1]] <= m[s[i]]) res += val;
            else res -= val;
        }
        return res;
    }
};

 

我們也可以每次跟前面的數字比較,如果小於等於前面的數字,我們先加上當前的數字,比如 "VI",第二個字母 'I' 小於第一個字母 'V',所以要加1。如果大於的前面的數字,我們加上當前的數字減去二倍前面的數字,這樣可以把在上一個迴圈多加數減掉,比如 "IX",我們在 i=0 時,加上了第一個字母 'I' 的值,此時結果res為1。當 i=1 時,我們發現字母 'X' 大於前一個字母 'I',這說明前面的1是要減去的,而由於我們前一步不但沒減,還多加了個1,所以此時要減去2倍的1,就是減2,所以才能得到9,整個過程是 res = 1 + 10 - 2 = 9,參見程式碼如下:

 

解法二:

class Solution {
public:
    int romanToInt(string s) {
        int res = 0;
        unordered_map<char, int> m{{'I', 1}, {'V', 5}, {'X', 10}, {'L', 50}, {'C', 100}, {'D', 500}, {'M', 1000}};
        for (int i = 0; i < s.size(); ++i) {
            if (i == 0 || m[s[i]] <= m[s[i - 1]]) res += m[s[i]];
            else res += m[s[i]] - 2 * m[s[i - 1]];
        }
        return res;
    }
};

 

類似題目:

Integer to Roman

 

參考資料:

https://leetcode.com/problems/roman-to-integer/

https://leetcode.com/problems/roman-to-integer/discuss/6547/Clean-O(n)-c%2B%2B-solution

https://leetcode.com/problems/roman-to-integer/discuss/6529/My-solution-for-this-question-but-I-don't-know-is-there-any-easier-way

 

相關文章