[LeetCode]Integer to Roman整数转罗马数字
2015-06-10 20:44
197 查看
作者:zf19921020
* 转载请注明出处*
Q:Integer to Roman
Des:Given an integer,convert it to a roman numeral.
Input is guaranteed to be within the range from 1 to 3999
罗马数字介绍
罗马数字表示法:1-9:I,II,III,IV,V,VI,VII,VIII,IX
10-90:X,XX,XXX,XL,L,LX,LXX,LXXX,XC
100-900:C,CC,CCC,CD,D,DC,DCC,DCCC,CM
也就是I表示1,V表示5,X表示10,L表示50,C表示100,D表示500,M表示1000
因此罗马数字能表示的最大数字为3999.
算法分析:将整型数字转化为罗马数字表示,而通过上面的罗马数字的表示可以看出,在不同的数字阶层:个位,十位,百位,千位,他们的数字表示模式都是相同的,因此,以下基于此给出两种不同的算法:
第一种是直接将所有罗马数字在各个阶层能表示的数值直接用String类型的数组保存起来,具体实现见代码:
LeetCode上A过的时间是42ms;该算法的优势在于短,特别短,两行代码。
第二种算法类似第一种,略有改动。基本思想是不管数字出于哪个阶层,将各个阶层的共同模式表示出来,之后利用字符串的替换方法将模式转化到对应的阶层,代码如下:
该代码在LeetCode上A过的时间是36ms。比之前的略有提升。其实两种算法的耗时都太长,所以在这里给出第三种算法。
算法的基本思想都是差不多的,不过这种算法是通过判定各个阶层和9与5之间的差值来进行相应的字符串操作。
代码源地址:https://leetcode.com/discuss/31704/c-fast-easy-to-read-understand-implementation
代码如下:
代码在LeetCode上A过的时间为16ms。
* 转载请注明出处*
Q:Integer to Roman
Des:Given an integer,convert it to a roman numeral.
Input is guaranteed to be within the range from 1 to 3999
罗马数字介绍
罗马数字表示法:1-9:I,II,III,IV,V,VI,VII,VIII,IX
10-90:X,XX,XXX,XL,L,LX,LXX,LXXX,XC
100-900:C,CC,CCC,CD,D,DC,DCC,DCCC,CM
也就是I表示1,V表示5,X表示10,L表示50,C表示100,D表示500,M表示1000
因此罗马数字能表示的最大数字为3999.
算法分析:将整型数字转化为罗马数字表示,而通过上面的罗马数字的表示可以看出,在不同的数字阶层:个位,十位,百位,千位,他们的数字表示模式都是相同的,因此,以下基于此给出两种不同的算法:
第一种是直接将所有罗马数字在各个阶层能表示的数值直接用String类型的数组保存起来,具体实现见代码:
string intToRoman(int num){ string romanPieces[{"","I","II","III","IV","V", "VI","VII","VIII","IX","","X","XX","XXX","XL", "L","LX","LXX","LXXX","XC","","C","CC","CCC", "CD","D","DC","DCC","DCCC","CM","","M","MM", "MMM","MMMM"}; return romanPieces[num/1000+30]+ romanPieces[(num/100)%10+20]+ romanPieces[(num/10)%10+10]+ romanPieces[num%10]; }
LeetCode上A过的时间是42ms;该算法的优势在于短,特别短,两行代码。
第二种算法类似第一种,略有改动。基本思想是不管数字出于哪个阶层,将各个阶层的共同模式表示出来,之后利用字符串的替换方法将模式转化到对应的阶层,代码如下:
//这一部分将对应的模式替换成相应的阶层 string replace_all(string &src, string replaceOfA, string replaceOfB = "", string replaceOfC = ""){ for (int index = 0; index < src.size(); ++index) switch (src.at(index)){ case 'A': src.replace(index, 1, replaceOfA); break; case'B': src.replace(index, 1, replaceOfB); break; case 'C': src.replace(index, 1, replaceOfC); break; } return src; } //这一部分将模式输入进去 string intToRoman(int num){ if (num > 3999 || num <= 0)return ""; vector<string> numCount = { "", "A", "AA", "AAA", "AB", "B", "BA", "BAA", "BAAA", "AC" }; int subtract = 1000; string str; while (subtract != 0){ string tmp = numCount.at(num / subtract); switch (subtract){ case 1000: replace_all(tmp, "M"); break; case 100: replace_all(tmp, "C", "D", "M"); break; case 10: replace_all(tmp, "X", "L", "C"); break; case 1: replace_all(tmp, "I", "V", "X"); break; } str += tmp; num = num%subtract; subtract /= 10; } return str; }
该代码在LeetCode上A过的时间是36ms。比之前的略有提升。其实两种算法的耗时都太长,所以在这里给出第三种算法。
算法的基本思想都是差不多的,不过这种算法是通过判定各个阶层和9与5之间的差值来进行相应的字符串操作。
代码源地址:https://leetcode.com/discuss/31704/c-fast-easy-to-read-understand-implementation
代码如下:
string intToRoman(int num){ string s = ""; for( ; num / 1000 ; ){ s += "M";num -=1000; } if( num >= 900 ){ s += "CM";num -=900; } if ( num >= 500 ){ s += "D";num -=500; } if ( num >= 400 ){ s += "CD";num -=400; } for( ; num / 100 ; ){ s += "C";num -=100; } if( num >= 90 ){ s += "XC";num -=90; } if ( num >= 50 ){ s += "L";num -=50; } if ( num >= 40 ){ s += "XL";num -=40; } for( ; num / 10 ; ){ s += "X";num -=10; } if ( num == 9 ){ s += "IX";return s; } if ( num >= 5 ){ s += "V";num -=5; } if ( num == 4 ){ s += "IV";return s; } for( ; num / 1 ; ){ s += "I";num -= 1; } return s; }
代码在LeetCode上A过的时间为16ms。
相关文章推荐
- Spark学习笔记:(一)入门 glance
- 无法连接mysql,请检查mysql是否已启动及用户密码是否设置正确
- [.net程序员必看]微软新动向之Android和IOS应用 visual studio 2015 Cordova[原创]
- 包含min函数的栈
- 获取导航栏,状态栏,标签栏的高度
- POJ 2125 Destroying The Graph 二分图最小点权覆盖 最小割
- Chrome rem bug
- 高程拾取器App上线
- 基于FPGA的双口RAM设计方法
- HDU 4998
- Android IllegalStateException: Fragment not attached to Activity
- 模式识别:k-均值聚类算法的研究与实现
- MySQL 数据(数据库)迁移
- HDU 1013 Digital Roots
- java发送http的get、post请求
- 顺时针打印矩阵
- Java正则表达式
- Flex 绘制圆形并填充图片
- HDU 1081 To The Max--DP--(最大子矩阵)
- 《C++ primer》第五版 第七章 笔记