首页 > 算法(附思维导图 + 全部解法)300题之13罗马数字转整数
头像
码农三少
发布于 2021-08-21 22:19
+ 关注

算法(附思维导图 + 全部解法)300题之13罗马数字转整数

零 标题:算法(leetode,附思维导图 + 全部解法)300题之(13)罗马数字转整数

导读:

一 题目描述

题目描述
题目描述
题目描述

二 解法总览(思维导图)

思维导图

三 全部解法

1 方案1

1)代码:

// 方案1 "状态机",一遍遍历。
var romanToInt = function(s) {
    const l = s.length;
    let index = 0,
        resNum = 0;

    // 1)不断往后走(“拉”),边界:index < l。
    // 核心:通过 if、else 等的组合“穷举所有的状态流转可能”。
    // 处理:根据最终流转出来的“状态”,变更 index 和 resNum 的值。
    while (index < l) {
        if (s[index] === 'M') {
            resNum += 1000;
        } else if (s[index] === 'D') {
            resNum += 500;
        } else if (s[index] === 'C') {
            if (s[index + 1] === 'D') {
                resNum += 400;
                // 边界:比其他普通情况多往后走1步,下同
                index++;
            } else if (s[index + 1] === 'M') {
                resNum += 900;
                index++;
            } else {
                resNum += 100;
            }
        } else if (s[index] === 'L') {
            resNum += 50;
        } else if (s[index] === 'X') {
            if (s[index + 1] === 'L') {
                resNum += 40;
                index++;
            } else if (s[index + 1] === 'C') {
                resNum += 90;
                index++;
            } else {
                resNum += 10;
            }
        } else if (s[index] === 'V') {
            resNum += 5;
        } else if (s[index] === 'I') {
            if (s[index + 1] === 'V') {
                resNum += 4;
                index++;
            } else if (s[index + 1] === 'X') {
                resNum += 9;
                index++;
            } else {
                resNum += 1;
            }
        }

        // 边界:不管怎么样都 ++ ,往往走1位
        index++;
    }

    // 2)返回上面所累计得到的 resNum 值
    return resNum;
}

2 方案2

1)代码:

// 方案2 建立所有的“数据映射集”,根据情况去分别做“加、减法”。
// 技巧:映射关系优先考虑hash这种数据结构(对应 JS的Map )
var romanToInt = function(s) {
    // 1)建立所有的“数据映射集”
    const l = s.length,
        map = new Map([
        ['I', 1],
        ['V', 5],
        ['X', 10],
        ['L', 50],
        ['C', 100],
        ['D', 500],
        ['M', 1000]
    ]);

    let index = 0,
        resNum = 0;

    // 2)根据情况去分别做“加、减法”(不断往后拉,边界: index < l)
    // III 可视为 0(初始化) + 1 + 1 + 1 = 3
    // IV 可视为 0(初始化) + 5(V) - 1(I,因为I放在V左边,故作减法) = 4
    while (index < l) {
        const tempNum = map.get(s[index]);
        if (index < l-1 && tempNum < map.get(s[index + 1])) {
            resNum -= tempNum;
        } else {
            resNum += tempNum;
        }

        index++;        
    }

    // 3)返回上面所累计得到的 resNum 值
    return resNum;
}

3 方案3

1)代码:

// 方案3 “预处理化”,根据预处理化后的“新罗马数值字符串”进行遍历。
// 技巧:“有序胜过无序”。
var romanToInt = function(s) {
    // 1)预处理化 —— 如将 'IV' 替换成 'a'、'IX' 替换成 'b' 等
    // a、b、c、d、e、f 分别表示 4、9、40、90、400、900。
    // 优化:链式调用。
    s = s.replace('IV', 'a')
        .replace('IX', 'b')
        .replace('XL', 'c')
        .replace('XC', 'd')
        .replace('CD', 'e')
        .replace('CM', 'f');

    // 2)通过 Map 定义全新的“数据映射集”。
    const l = s.length,
        map = new Map([
        ['I', 1],
        ['V', 5],
        ['X', 10],
        ['L', 50],
        ['C', 100],
        ['D', 500],
        ['M', 1000],
        ['a', 4],
        ['b', 9],
        ['c', 40],
        ['d', 90],
        ['e', 400],
        ['f', 900]
    ]);

    let index = 0,
        resNum = 0;

    // 3)根据替换后、“新罗马数值字符串”进行遍历,结合map去变更 resNum 值。
    while (index < l) {
        resNum += map.get(s[index]);
        index++;
    }

    // 4)返回上面所累计得到的 resNum 值。
    return resNum;
}

全部评论

(0) 回帖
加载中...
话题 回帖

推荐话题

相关热帖

热门推荐