【LeetCode-简单】13. 罗马数字转整数(详解)

题目

罗马数字包含以下七种字符: I, V, X, LCD 和 M

例如, 罗马数字 2 写做 II ,即为两个并列的 1 。12 写做 XII ,即为 X + II 。 27 写做  XXVII, 即为 XX + V + II 。

通常情况下,罗马数字中小的数字在大的数字的右边。但也存在特例,例如 4 不写做 IIII,而是 IV。数字 1 在数字 5 的左边,所表示的数等于大数 5 减小数 1 得到的数值 4 。同样地,数字 9 表示为 IX。这个特殊的规则只适用于以下六种情况:

I 可以放在 V (5) 和 X (10) 的左边,来表示 4 和 9。(情况1、2)
X 可以放在 L (50) 和 C (100) 的左边,来表示 40 和 90。 (情况3、4)
C 可以放在 D (500) 和 M (1000) 的左边,来表示 400 和 900。(情况5、6)
给定一个罗马数字,将其转换成整数。


题目链接:https://leetcode.cn/problems/roman-to-integer

示例

 

 解法1:字符串转数组(自己写的方法)

思路

正常的情况:直接从左往右相加。

特殊的6中情况:1、2 情况,相加之后再-2;

                            3、4情况 ,相加之后再-20;

                            5、6情况,相加之后再-200;

步骤

1.使用s.toCharArray() 方法,将字符串转换成对应的字符数组。

2.再遍历字符串数组,将字幕替换成对应的数字。

3.再次遍历数组,直接求和。

4.如果前一个数 < 后一个数 ,说明是特殊情况,对应处理即可。

    //排除特殊的情况:特殊情况就是比正常情况-2、-20、-200
    //正常情况就是:从左往右加法
    public int romanToInt(String s) {
        char arr[]=s.toCharArray();
        int sum = 0;
        for (int i=0;i<arr.length;i++){
            switch (arr[i]){
                case 'I':arr[i]=1;break;
                case 'V':arr[i]=5;break;
                case 'X':arr[i]=10;break;
                case 'L':arr[i]=50;break;
                case 'C':arr[i]=100;break;
                case 'D':arr[i]=500;break;
                case 'M':arr[i]=1000;break;
            }
        }
        for (int i=0;i<arr.length;i++){
            sum = sum + arr[i];
            if ( i < arr.length-1 && arr[i]<arr[i+1]){
                if (arr[i]==1)
                    sum = sum -2;
                if (arr[i]==10)
                    sum = sum -20;
                if (arr[i]==100)
                    sum = sum -200;
            }
        }

        return sum;
    }

效果

解法2:对解法1 改善一下

思路

解法1特殊情况那里太啰嗦了

上面特殊情况是要做判断,更具情况来对结果减去多少

看了网上的解法,有更好的:如果前一个比后一个小,那么 直接将符号改成负号即可

妙啊! 例如 IV 就变成了 -1 + 5 = 4

这样代码简洁一些,但效果差不多

     //排除特殊的情况:特殊情况就是把 符号改成负号即可
    //正常情况就是:从左往右加法
    public int romanToInt(String s) {
        char arr[]=s.toCharArray();
        int sum = 0;
        for (int i=0;i<arr.length;i++){
            switch (arr[i]){
                case 'I':arr[i]=1;break;
                case 'V':arr[i]=5;break;
                case 'X':arr[i]=10;break;
                case 'L':arr[i]=50;break;
                case 'C':arr[i]=100;break;
                case 'D':arr[i]=500;break;
                case 'M':arr[i]=1000;break;
            }
        }
        for (int i=0;i<arr.length;i++){
            if ( i < arr.length-1 && arr[i]<arr[i+1]){
                sum = sum -arr[i];
            }else  sum = sum + arr[i];
        }

        return sum;
    }

解法3:用哈希表(力扣官方解法)

思路

与解法2相同,只是解法2用的是 字符串数组,而这里用的是哈希表

class Solution {
    Map<Character, Integer> symbolValues = new HashMap<Character, Integer>() {{
        put('I', 1);
        put('V', 5);
        put('X', 10);
        put('L', 50);
        put('C', 100);
        put('D', 500);
        put('M', 1000);
    }};

    public int romanToInt(String s) {
        int ans = 0;
        int n = s.length();
        for (int i = 0; i < n; ++i) {
            int value = symbolValues.get(s.charAt(i));
            if (i < n - 1 && value < symbolValues.get(s.charAt(i + 1))) {
                ans -= value;
            } else {
                ans += value;
            }
        }
        return ans;
    }
}

效果

还不如我写的那个,但我没有想到用哈希表。 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值