NOIP2008解题报告
2008年的题目相对比较简单,都不是很麻烦,认真写就能写对。
第一题:
题目大意:给出一个小写单词,求出最少出现的字母的出现次数和最多出现的字母的出现次数。
解题过程:直接hash模拟就好。
第二题:
给你n根火柴棍,你可以拼出多少个形如“A+B=C”的等式?等式中的A、B、C是用火柴棍拼出的整数(若该数非零,则最高位不能是0)。用火柴棍拼数字0-9的拼法如图所示:
注意:
1. 加号与等号各自需要两根火柴棍
2. 如果A≠B,则A+B=C与B+A=C视为不同的等式(A、B、C>=0)
3. n根火柴棍必须全部用上。n<=24
解题过程:
1.一开始看到A+B=C,想到直接枚举,但是也有可能出现1111+0=1111的情况。。因此不好确定枚举的范围。
2.于是决定用dfs两个加数,然后看剩下的火柴能否拼出和。对于任意一个数,可以在它的末尾加上0~9的数字,代价是相应需要的火柴数量,这样保证不会重复。有个小优化,不过我懒得写了,就是令第一个数不比第二个数大,搜到结果时,如果第一个数和第二个数相等,ans+1,否则ans+2。这样可以减少一半的搜索量,不过数据如此小,不写也无所谓。
第三题:
题目大意:就是方格取数的变形,变成了2个人一起走,且另外一个人走过的地方不能再走。
解题过程:
和2000年方格取数那题几乎一样。只不过00年的是走过的地方可以再走,只不过得到的分数为0,因此只要在00年那题的动态规划方程里改一下,如果i=j且没有到达终点的时候,F[k][i][j]=-inf. F[k][i][j]表示走了k步,第一个人向下走了i步,第二个人向下走了j步的最大得分。
动态规划的部分代码:
1 for (int k=1;k<=n+m-2;k++) 2 for (int i=0;i<n;i++) 3 for (int j=0;j<n;j++) 4 { 5 if (i==j&&k!=n+m-2) 6 { 7 g[k][i][j]=-210000000; 8 continue; 9 } 10 if (i!=0 && j!=0) 11 g[k][i][j]=ma(g[k][i][j],g[k-1][i-1][j-1]); 12 if (i!=0 && j!=k) 13 g[k][i][j]=ma(g[k][i][j],g[k-1][i-1][j]); 14 if (i!=k && j!=k) 15 g[k][i][j]=ma(g[k][i][j],g[k-1][i][j]); 16 if (i!=k && j!=0) 17 g[k][i][j]=ma(g[k][i][j],g[k-1][i][j-1]); 18 g[k][i][j]+=a[i+1][1+k-i]+a[j+1][1+k-j]; 19 }
第四题:
Tom最近在研究一个有趣的排序问题。如图所示,通过2个栈S1和S2,Tom希望借助以下4种操作实现将输入序列升序排序。
操作a
如果输入序列不为空,将第一个元素压入栈S1
操作b
如果栈S1不为空,将S1栈顶元素弹出至输出序列
操作c
如果输入序列不为空,将第一个元素压入栈S2
操作d
如果栈S2不为空,将S2栈顶元素弹出至输出序列
如果一个1~n的排列P可以通过一系列操作使得输出序列为1,2,…,(n-1),n,Tom就称P是一个“可双栈排序排列”。例如(1,3,2,4)就是一个“可双栈排序序列”,而(2,3,4,1)不是。下图描述了一个将(1,3,2,4)排序的操作序列:<a,c,c,b,a,d,d,b>
当然,这样的操作序列有可能有几个,对于上例(1,3,2,4),<a,c,c,b,a,d,d,b>是另外一个可行的操作序列。Tom希望知道其中字典序最小的操作序列是什么。
解题过程:
1.一开始手工模拟了一下,感觉可以贪心。如果当前元素比栈顶元素大,那么肯定不能放到栈顶。(有点类似汉诺塔。)贪心策略如下:能往A栈放的就尽量往A栈放,否则放B栈,如果两个都没法放,就输出无解。如果当前元素就是下一个要到输出队列的元素,那么进栈A然后出来,也就是操作’a‘+’b',然后用一个while把该出来的都出来。。很快写好,自己写栈也才1000b左右,但是提交只能过5个点。。考试来不及这样写也不错。。
2.正解:参考博客http://blog.csdn.net/kqzxcmh/article/details/9566813 写的非常清晰,代码也简洁易懂,短小精悍。。
首先根据栈的性质,S[i],S[j]两个元素不能进入同一个栈 <=> 存在k,满足i<j<k,使得S[k]<S[i]<S[j]. 不会证明,但绝对正确。。简单点说,就是对于入栈序列的某个元素,在它前面的且比它大的元素必须是递增的(从右往左看递增,从左往右递减。)这样好理解,但是编程时还是用存在k,满足i<j<k,使得S[k]<S[i]<S[j]. 来写。
因此只要找出所有不能在同时在同一个栈的入栈序列出现的数对(i,j),让前面的元素尽量进A栈。这实际上是一个判断二分图的过程,用染色的方法判断,如果不是二分图必然无解。如果是,模拟出栈即可。
我的代码:
1 #include<iostream> 2 #include<cstdio> 3 using namespace std; 4 5 char ans[2014]; 6 int n,a[1010],stack1[1010],stack2[1010],top1,top2,cur; 7 int p[1010],q[1010]; 8 int g[1010],color[1010],edge[1010][1010],flag=1; 9 10 void pop() 11 { 12 while (stack1[top1]==cur||stack2[top2]==cur) 13 { 14 if (stack1[top1]==cur) 15 { 16 cur++; 17 top1--; 18 printf("b "); 19 } 20 else 21 { 22 cur++; 23 top2--; 24 printf("d "); 25 } 26 } 27 } 28 29 void dfs(int x,int c) 30 { 31 if (!flag) 32 return; 33 color[x]=c; 34 for (int i=1;i<=n;i++) 35 if (edge[x][i]) 36 { 37 if (color[i]==c) 38 { 39 flag=0; 40 return; 41 } 42 if (!color[i]) 43 dfs(i,3-c); 44 } 45 } 46 47 void solve() 48 { 49 g[n+1]=210000; 50 for (int i=n;i>=1;i--) 51 g[i]=a[i]<g[i+1]? a[i]:g[i+1]; 52 for (int i=1;i<n;i++) 53 for (int j=i+1;j<=n;j++) 54 if (a[i]<a[j]&&g[j+1]<a[i]) 55 edge[i][j]=edge[j][i]=1; 56 for (int i=1;i<=n;i++) 57 if (!color[i]) 58 dfs(i,1); 59 if (!flag) 60 { 61 cout<<0<<endl; 62 return; 63 } 64 cur=1; 65 stack1[0]=stack2[0]=210000; 66 for (int i=1;i<=n;i++) 67 { 68 if (color[i]==1) 69 { 70 printf("a "); 71 stack1[++top1]=a[i]; 72 } 73 else 74 { 75 printf("c "); 76 stack2[++top2]=a[i]; 77 } 78 pop(); 79 } 80 pop(); 81 } 82 83 int main() 84 { 85 //freopen("in.txt","r",stdin); 86 //freopen("out.txt","w",stdout); 87 cin >> n; 88 for (int i=1;i<=n;i++) 89 scanf("%d",&a[i]); 90 solve(); 91 return 0; 92 }