题目描述
一类书的序言是以罗马数字标页码的。传统罗马数字用单个字母表示特定的数值,一下是标准数字表:
I 1 L 50 M 1000
V 5 C 100
X 10 D 500
最多3个可以表示为10n的数字(I,X,C,M)可以连续放在一起,表示它们的和:
III=3
CCC=300
可表示为5x10n的字符(V,L,D)从不连续出现。
除了下一个规则,一般来说,字符以递减的顺序接连出现:
CCLXVIII = 100+100+50+10+5+1+1+1 = 268
有时,一个可表示为10^n的数出现在一个比它大的数前(I在V或X前面,X在L或C前面,等等)。在这种情况下,数值等于后面的那个数减去前面的那个数:
IV = 4
IX = 9
XL = 40
像XD, IC, 和XM这样的表达是非法的,因为前面的数比后面的数小太多。对于XD(490的错误表达),可以写成 CDXC; 对于IC(99的错误表达),可以写成XCIX; 对于XM(990的错误表达),可以写成CMXC。
给定N(1 <= N < 3,500), 序言的页码数,请统计在第1页到第N也中,有几个I出现,几个V出现,等等 (从小到大的顺序)。不要输出并没有出现过的字符。
比如N = 5, 那么页码数为: I, II, III, IV, V. 总共有7个I出现,2个V出现。
输入
一个整数N。
输出
每行一个字符和一个数字k,表示这个字符出现了k次。字符必须按数字表中的递增顺序输出。
样例输入
5
样例输出
I 7
V 2
题意分析:这道题不要误以为只输出I和V,其次就是输出的顺序是按给定的字符所代表的数字大小按从小到大输出的
解题思路:就是纯思维模拟,不过还是有点规律的自己列一个从1到9所代表的序言页码,然后在列一个10 20 30 40 50 60 70 80 90的序言页码,在列一行100 200 300 400 500 600 700 800 900对应的序言页码就会找到一点规律。
#include<stdio.h>
#include<math.h>
#include<map>
using namespace std;
int a[2][10]={{0,1,2,3,1,0,1,2,3,1},
{0,0,0,0,1,1,1,1,1,0}};
int main()
{
map<char,int> m;//记录出现的次数
map<int,char> n;//记录顺序
int t,i,j,k,k1;
while(~scanf("%d",&t))
{
m['I']=0;m['V']=0;m['X']=0;
m['L']=0;m['C']=0;m['D']=0;m['M']=0;
n[1]='I';n[2]='V';n[3]='X';
n[4]='L';n[5]='C';n[6]='D';n[7]='M';
for(i=1;i<=t;i++)
{
k1=i;
k=(int)log10(i)+1;
if(k>3)
{
m['M']+=a[0][i/1000];
k1-=i/1000*1000;
j=k1;
}
else
j=k1;
if(j>=100&&j<1000)
{
m['C']+=a[0][j/100];
m['D']+=a[1][j/100];
if(j>900&&j<1000)
m['M']++;
j-=j/100*100;
}
m['I']+=a[0][j%10];
m['V']+=a[1][j%10];
if(j>=10&&j<100)
{
if(j%10==9)
m['X']++;
m['X']+=a[0][j/10];
m['L']+=a[1][j/10];
if(j>90)
m['C']++;
}
if(j==9)
m['X']++;
if(j==90)
m['C']++;
if(i==900||i==1999||i==2900)//这几个数是需要自己慢慢算出来的
m['M']++;
}
for(i=1;i<=7;i++)
{
if(m[n[i]]!=0)
printf("%c %d\n",n[i],m[n[i]]);
}
}
return 0;
}