Given an integer, convert it to a roman numeral.
Input is guaranteed to be within the range from 1 to 3999.
意思就是:
给出一个整数 num( 0<=num<=3999),返回其对应的罗马数字表示;
罗马数字有:
|I |V |X |L |C |D |M |
|--|--|--|--|--|--|--|
|1 |5 |10|50|100|500|1000|
这个方式不知道为什么很慢,是我第一个想出的几乎相当于暴力的方法。
class Solution {
public:
string intToRoman(int num) {
if(num<=0||num>3999)return string();
map<int,string> one,two,three,four; //这里使用的是 Map
one[0]=""; two[0]=""; three[0]="";
one[1]="I"; two[1]="X"; three[1]="C"; four[1]="M";
one[2]="II"; two[2]="XX"; three[2]="CC"; four[2]="MM";
one[3]="III"; two[3]="XXX"; three[3]="CCC"; four[3]="MMM";
one[4]="IV"; two[4]="XL"; three[4]="CD";
one[5]="V"; two[5]="L"; three[5]="D";
one[6]="VI"; two[6]="LX"; three[6]="DC";
one[7]="VII"; two[7]="LXX"; three[7]="DCC";
one[8]="VIII"; two[8]="LXXX"; three[8]="DCCC";
one[9]="IX"; two[9]="XC"; three[9]="CM";
int mod=0,idx=1;
string rs="",tmp="";
while(num>0){
mod=num%10;
switch(idx){
case 1:{
tmp=one[mod];
rs=tmp+rs;
break;
}
case 2:{
tmp=two[mod];
rs=tmp+rs;
break;
}
case 3:{
tmp=three[mod];
rs=tmp+rs;
break;
}
case 4:{
tmp=four[mod];
rs=tmp+rs;
break;
}
}
num=num/10;
++idx;
}
return rs;
}
};
果然,改进后,运行时间从原来的 143ms 变成了 68ms ,这是第一个改进:
//原来的 map
map<int,string> one,two,three,four; //这里使用的是 Map
//改进后
string* one=new string[10],*two=new string[10],*three=new string[10],*four=new string[4];
方法1和2中,使用了很大的空间来存储全部的罗马特殊数字,增加了空间复杂性,所以我就想,能不能不存储这些特殊数字,只存储最基本的罗马基数数字 IVXLCDM
,然后根据当前数字进行判断,生成对应的罗马数字。下面是这一思想的代码:
class Solution {
public:
string intToRoman(int num) {
if(num<=0||num>3999)return string();
string roman="IVXLCDM"; //存储基数
string rs="",tmp="";
int ndx=0,mod=0; // ndx 用来记录当前基数(即第n位对应的最小罗马数字坐标)
while(num>0){
mod=num%10; //求余数
if(ndx==6){ //千位只有 M
rs=string(mod,'M')+rs;
break;
}
if(mod==0){
tmp="";
}else if(mod<4){
tmp=string(mod,roman[ndx]);
}else if(mod==4){
tmp=string(1,roman[ndx])+string(1,roman[ndx+1]);
}else if(mod<9){
tmp=string(1,roman[ndx+1])+string(mod-5,roman[ndx]);
}else if(mod==9){
tmp=string(1,roman[ndx])+string(1,roman[ndx+2]);
}
rs=tmp+rs;
ndx+=2; //基数向前进2个
num/=10;
}
return rs;
}
};
这一思想中,有两点是需要考虑的:
M
;4 , 9
很特殊;对于数字 4 来说,如果当前基数坐标为 ndx ,则其对应的罗马数字为:
string(1,roman[ndx])+string(1,roman[ndx+1]);
比如当前位为十位,则 ndx=2; roman[ndx]='X'; roman[ndx+1]='L' ;
对应罗马数字为: XL
对于数字 9 来说,对应罗马数字为:
string(1,roman[ndx])+string(1,roman[ndx+2]);
比如当前位为十位,则 ndx=2; roman[ndx]='X'; roman[ndx+2]='C' ;
对应罗马数字为: XC
roman[ndx+1]
和 mode-5 个连续基数;对于代码的进行,方法3中会产生大量的string临时变量,因此这里尽量的使用了迭代器和 string 的insert方法进行处理。代码运行效率得以再次提高:48ms
class Solution {
public:
string intToRoman(int num) {
if(num<=0||num>3999)return string();
char roman[]="IVXLCDM";
string rs="";
int ndx=0,mod=0;
while(num>0){
mod=num%10;
if(ndx==6){
rs=string(mod,'M')+rs;
break;
}
if(mod<4){
rs=string(mod,roman[ndx])+rs;
}else if(mod==4){
rs.insert(rs.begin(),roman[ndx+1]);
rs.insert(rs.begin(),roman[ndx]);
// rs=string(1,roman[ndx])+string(1,roman[ndx+1])+rs;
}else if(mod<9){
// rs=string(1,roman[ndx+1])+string(mod-5,roman[ndx])+rs;
rs=string(mod-5,roman[ndx])+rs;
rs.insert(rs.begin(),roman[ndx+1]);
}else if(mod==9){
rs.insert(rs.begin(),roman[ndx+2]);
rs.insert(rs.begin(),roman[ndx]);
// rs=string(1,roman[ndx])+string(1,roman[ndx+2])+rs;
}
ndx+=2;
num/=10;
}
return rs;
}
};
这是LeetCode上网友的答案,感觉跟简洁,同时算法也很容易理解,和这里第一种的思想很像:
class Solution {
public:
string intToRoman(int num) {
string M[] = {"", "M", "MM", "MMM"};
string C[] = {"", "C", "CC", "CCC", "CD", "D", "DC", "DCC", "DCCC", "CM"};
string X[] = {"", "X", "XX", "XXX", "XL", "L", "LX", "LXX", "LXXX", "XC"};
string I[] = {"", "I", "II", "III", "IV", "V", "VI", "VII", "VIII", "IX"};
return M[num/1000] + C[(num%1000)/100] + X[(num%100)/10] + I[num%10];
}
};
这个是网友的答案:
class Solution {
public:
string intToRoman(int num) {
// IMPORTANT: Please reset any member data you declared, as
// the same Solution instance will be reused for each test case.
string result = "";
int base[] = {1000,500,100,50,10,5,1,0};
char baseC[] = {'M','D','C','L','X','V','I'};
int basen = 0;
while(num) {
if(basen%2 == 0 && num/base[basen] == 4) {
result += baseC[basen];
result += baseC[basen-1];
num -= base[basen] * 4;
} else if(num >= base[basen]) {
result += baseC[basen];
num -= base[basen];
} else if(basen%2 == 0 && num / base[basen+2] == 9) {
result += baseC[basen+2];
result += baseC[basen];
num -= base[basen+2]*9;
} else {
basen++;
}
}
return result;
}
};
网友给出的思路是这样的:
个人觉得这个算法有点复杂!不过还是挺高效的。这里学习一下思想。
同系列:LeetCodesOJhttp://www.cnblogs.com/lomper/tag/LeetCodesOJ/