109. Magic of David Copperfield II 每个数据时间限制: 0.50 sec.
著名的魔术师大卫科波菲尔喜欢表演下面的魔术:一个N行N列不同图片的矩阵出现在大屏幕上,我们给所有的图片这样命名:
每一个参与的观众被要求将手指放在左上方的图片上(即编号为1的图片),魔术师开始了:魔术师告诉观众在图片上移动k次(移动是把手指放到上下左右相邻的图片上,如果那儿有图片的话),然后他(魔术师)的手微微一指(指向一些图片)并说:“你不在这里”,然后……是真的!你的手指没有指向任何一个被删除的图片(指向的图片)然后再来一次,他告诉观众再移动K2次……以此类推。在最后,他删除到只剩最后一个图片了,然后胜利地微笑着宣布“我抓到你了!”(掌声)。
输入: 输入文件包含一个整数N (1<N<101).
输出: 你的程序需要像下面这样输出数字:
样例输入: 3
样例输出: 3 1 3 7 9 5 2 4 6 8 |
================================华丽的分割线 ================================
(待添加题解 2011年6月10日 19:10:51)
2011年6月10日 21:59:17。
做了这个题目我清楚的认识到了,在考试时做题目要冷静,这种题目关乎矩阵的就先画一个图,然后慢慢分析!也学一个乖,这种在矩阵上相邻的位置移动的,就要分奇数格和偶数格。
http://www.docin.com/p-212303829.html,很不错的题解!特别是那几幅图!还有,我的代码是参考一个大牛的,本来我是按照那个题解的方案二写的,但是出现的问题就是最终k>300,然后就WA了!
这里的优化技巧是,第一次移动就是N,然后从(3,n)为左上角,斜着的一条线下面的整个三角形都可以放在第一次移动,这样直接减少了一般的移动次数,然后再用上面那个链接的方案二。
#include <stdio.h>
#include <stdlib.h>
int tot;
int n;
//从右上角(x, y)向左下角输出
void output(int x, int y, int end)
{
if(x > n || y < 1){
if(end){
printf("\n");
}
return;
}
printf(" %d", (x - 1) * n + y);
output(x + 1, y - 1, end);
}
int main(int argc, char **argv)
{
int i, k;
scanf("%d", &n);
//有点麻烦的特殊情况, 直接..作弊算了
if(n == 2){
printf("3 4\n5 2 3\n");
return 0;
}
printf("%d", n);
for(i = 3; i <= n; i++){
output(i, n, 0);
}
if(n & 1){
k = n + 2;
}else{
k = n + 1;
}
printf("\n%d", k);
output(2, n, 1);
for(i = n; i >= 2; i--){
k += 2;
printf("%d", k);
output(1, i, 1);
}
return 0;
}