USACO 2.2 Preface Numbering 【实质是分治思想】
一类书的序言是以罗马数字标页码的。传统罗马数字用单个字母表示特定的数值,以下是标准数字表:
I 1 V 5 X 10 L 50 C 100 D 500 M 1000
最多3个同样的可以表示为10n的数字(I,X,C,M)可以连续放在一起,表示它们的和:
III=3 CCC=300
可表示为5x10n的字符(V,L,D)从不连续出现。
除了下一个规则,一般来说,字符以递减的顺序接连出现:
CCLXVIII = 100+100+50+10+5+1+1+1 = 268
有时,一个可表示为10n的数出现在一个比它大1级或2级的数前(I在V或X前面,X在L或C前面,等等)。在这种情况下,数值等于后面的那个数减去前面的那个数:
IV = 4 IX = 9 XL = 40
一个数 用罗马数字来表示 有且仅有一种 而且不能复合嵌套使用(比如I是1 X是10 有人可能要说 IXL就能表示50-10-1 但是IXL绝对不能用来表达39 ) (那么39用什么来表示呢 XXXIX是唯一 而且正确的选择- -)
像XD, IC, 和XM这样的表达是非法的,因为前面的数比后面的数小太多。对于XD(490的错误表达),可以写成 CDXC; 对于IC(99的错误表达),可以写成XCIX; 对于XM(990的错误表达),可以写成CMXC。 90 写成 XC 而不是 LXL, 因为 L 后面的 X 意味着后继标记是 X 或者更小 (不管怎样,可能吧)(等同于阿拉伯数字 每位 数字分别表示)。
给定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
4 I 1
1 V 5
5 X 10
2 L 50
6 C 100
3 D 500
7 M 1000
1 #include <iostream> 2 #include <cstdio> 3 #include <cstring> 4 #include <algorithm> 5 using namespace std; 6 int n; 7 int a[8]={0}; 8 void f(int x) 9 { 10 int t=0; 11 while(x>0) 12 { 13 t++;//相当于k,记录到了第几位,到时候*10^k 14 int xx=x%10; 15 if(xx==9) 16 { 17 a[t+3]++; // IX 中的" I " 18 a[t+4]++; // IX 中的" X " 19 } 20 else 21 { 22 if(xx>=5) 23 { 24 a[t]++; // " V " 25 xx-=5; 26 } 27 if(xx==4) 28 { 29 a[t]++; 30 a[t+3]++; 31 } 32 else 33 a[t+3]+=xx; 34 } 35 x/=10; // 一位一位剥掉它 36 } 37 } 38 39 40 int main() 41 { 42 freopen("preface.in","r",stdin); 43 freopen("preface.out","w",stdout); 44 cin>>n; 45 for(int i=1;i<=n;i++) 46 f(i); 47 if(a[4]) cout<<'I'<<" "<<a[4]<<endl; 48 if(a[1]) cout<<'V'<<" "<<a[1]<<endl; 49 if(a[5]) cout<<'X'<<" "<<a[5]<<endl; 50 if(a[2]) cout<<'L'<<" "<<a[2]<<endl; 51 if(a[6]) cout<<'C'<<" "<<a[6]<<endl; 52 if(a[3]) cout<<'D'<<" "<<a[3]<<endl; 53 if(a[7]) cout<<'M'<<" "<<a[7]<<endl; 54 return 0; 55 }