leecodeアルゴリズム「13.ローマ数字回転整数」は注釈が詳しく、簡単明瞭である.


leecodeアルゴリズム「13.ローマ数字回転整数」は注釈が詳しく、簡単明瞭である.
原題の内容
ローマ数字には、I、V、X、L、C、D、Mの7文字が含まれています.
文字値I 1 V 5 X 10 L 50 C 100 D 500 M 1000は、例えば、ローマ数字2をIIと書くと、2つの並列の1となる.12はXIIと書きます.つまりX+IIです.27 XXVIIと書くと、XX+V+IIとなります.
通常、ローマ数字の中で小さい数字は大きな数字の右側にあります.しかし、例えば4はIIIではなくIVと書くという特例もある.数字1は数字5の左側にあり、表す数は大数5から小数1を減らした数値4に等しい.同様に、数字9はIXとして表される.この特殊なルールは、次の6つの場合にのみ適用されます.
IはV(5)とX(10)の左側に置いて4と9を表すことができる.XはL(50)とC(100)の左側に置いて40と90を表すことができる.CはD(500)とM(1000)の左側に置いて400と900を表すことができる.ローマ数字を指定し、整数に変換します.入力は1~3999の範囲であることを確認します.
例1:
入力:III出力:3例2:
入力:IV出力:4例3:
入力:IX出力:9例4:
入力:「LVIII」出力:58解釈:L=50,V=5,III=3.例5:
入力:「MCMXCIV」出力:1994解釈:M=1000,CM=900,XC=90,IV=4.
ソース:力ボタン(LeetCode)リンク:https://leetcode-cn.com/problems/roman-to-integer著作権はインターネットの所有に帰属する.商業転載は公式の授権に連絡してください.非商業転載は出典を明記してください.
審査問題:解法一
この問題はやはり簡単な第一歩です.mapの第2ステップを構築します.まず配列長が1であるかどうかを判断し,1であれば直接第3ステップに戻る:ループは2つを比較するたびに私が考えている構想は2つを取り出して表を合わせることである
コード:
    public static void main(String[] args) {
        System.out.println(romanToInt("MDCXCV"));
    }

    public static int romanToInt(String s) {
        //   map
        Map<Character, Integer> map = new HashMap<>(6);
        map.put('I', 1);
        map.put('V', 5);
        map.put('X', 10);
        map.put('L', 50);
        map.put('C', 100);
        map.put('D', 500);
        map.put('M', 1000);
        int result = 0;
        //        1    map       
        if (s.length() == 1) {
            return map.get(s.charAt(0));
        }
        //            
        for (int i = 0; i < s.length() - 1; i++) {
            //               ,      
            if (map.get(s.charAt(i + 1)) <= map.get(s.charAt(i))) {
                //      i                    ;
                if (i == s.length() - 2) {
                    result = map.get(s.charAt(i + 1)) + result;
                }
                result = map.get(s.charAt(i)) + result;
            } else {
                result = result + map.get(s.charAt(i + 1)) - map.get(s.charAt(i));
                i++;
                //      i                    ;
                if (i == s.length() - 2) {
                    result = map.get(s.charAt(i + 1)) + result;
                }
            }
        }
        return result;
    }

審査問題:解法二
まず、すべての組合せ可能性をハッシュ・テーブルにリストして追加し、文字列を巡回します.組合せは2つしかないため、1つは1文字、1つは2文字で、2つの文字が1文字よりも優先され、2つの文字の組合せがハッシュ・テーブルに存在するかどうかを判断し、存在する場合は結果ansに値を取り出し、2文字後ろに移動します.存在しない場合は現在の1文字が存在するか否かを判断し、存在する場合は結果ansに値を取り出し、1文字を後方にシフトしてループ終了して結果ansに戻る
class Solution {
    public int romanToInt(String s) {
        Map<String, Integer> map = new HashMap<>();
        map.put("I", 1);
        map.put("IV", 4);
        map.put("V", 5);
        map.put("IX", 9);
        map.put("X", 10);
        map.put("XL", 40);
        map.put("L", 50);
        map.put("XC", 90);
        map.put("C", 100);
        map.put("CD", 400);
        map.put("D", 500);
        map.put("CM", 900);
        map.put("M", 1000);
        
        int ans = 0;
        for(int i = 0;i < s.length();) {
            if(i + 1 < s.length() && map.containsKey(s.substring(i, i+2))) {
                ans += map.get(s.substring(i, i+2));
                i += 2;
            } else {
                ans += map.get(s.substring(i, i+1));
                i ++;
            }
        }
        return ans;
    }
}

審査問題:解法三
この考え方も非常に良いコードに注釈が付けられています
public static class Solution1 {
 public int romanToInt(String s) {
  Map<Character, Integer> map = new HashMap();
  map.put('I', 1);
  map.put('V', 5);
  map.put('X', 10);
  map.put('L', 50);
  map.put('C', 100);
  map.put('D', 500);
  map.put('M', 1000);

  int result = 0;
  for (int i = 0; i < s.length(); i++) {
   if (i > 0 && map.get(s.charAt(i)) > map.get(s.charAt(i - 1))) {
   //            2  2 * map.get(s.charAt(i - 1)
   //     a+B-2a=B-a
   //                      。
    result += map.get(s.charAt(i)) - 2 * map.get(s.charAt(i - 1));
   } else {
    result += map.get(s.charAt(i));
   }
  }
  return result;
 }
}