[NOIP2009]靶形数独 题解
Posted Forever_goodboy
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[NOIP2009]靶形数独 题解相关的知识,希望对你有一定的参考价值。
407. [NOIP2009] 靶形数独
★★ 输入文件:sudoku.in
输出文件:sudoku.out
简单对比
时间限制:5 s 内存限制:128 MB
【问题描述】
小城和小华都是热爱数学的好学生,最近,他们不约而同地迷上了数独游戏,好胜的他们想用数独来一比高低。但普通的数独对他们来说都过于简单了,于是他们向 Z博士请教,Z 博士拿出了他最近发明的“靶形数独” ,作为这两个孩子比试的题目。
靶形数独的方格同普通数独一样,在 9 格宽×9 格高的大九宫格中有 9 个 3 格宽×3 格高的小九宫格(用粗黑色线隔开的) 。在这个大九宫格中,有一些数字是已知的,根据这些数字,利用逻辑推理,在其他的空格上填入 1到 9 的数字。每个数字在每个小九宫格内不能重复出现,每个数字在每行、每列也不能重复出现。但靶形数独有一点和普通数独不同,即每一个方格都有一个分值,而且如同一个靶子一样,离中心越近则分值越高。 (如图)
上图具体的分值分布是:最里面一格(黄色区域)为 10 分,黄色区域外面的一圈(红色区域)每个格子为 9 分,再外面一圈(蓝色区域)每个格子为 8分,蓝色区域外面一圈(棕色区域)每个格子为 7分,最外面一圈(白色区域)每个格子为 6 分,如上图所示。比赛的要求是:每个人必须完成一个给定的数独(每个给定数独可能有不同的填法) ,而且要争取更高的总分数。而这个总分数即每个方格上的分值和完成这个数独时填在相应格上的数字的乘积的总和。如图,在以下的这个已经填完数字的靶形数独游戏中,总分数为 2829。游戏规定,将以总分数的高低决出胜负。
由于求胜心切,小城找到了善于编程的你,让你帮他求出,对于给定的靶形数独,能够得到的最高分数。
【输入】
输入文件名为 sudoku.in。
一共 9 行。每行 9 个整数(每个数都在 0—9 的范围内) ,表示一个尚未填满的数独方格,未填的空格用“0”表示。每两个数字之间用一个空格隔开。
【输出】
输出文件 sudoku.out共 1行。
输出可以得到的靶形数独的最高分数。如果这个数独无解,则输出整数-1。
【输入输出样例 1】
sudoku.in
7 0 0 9 0 0 0 0 1
1 0 0 0 0 5 9 0 0
0 0 0 2 0 0 0 8 0
0 0 5 0 2 0 0 0 3
0 0 0 0 0 0 6 4 8
4 1 3 0 0 0 0 0 0
0 0 7 0 0 2 0 9 0
2 0 1 0 6 0 8 0 4
0 8 0 5 0 4 0 1 2
sudoku.out
2829
【输入输出样例 2】
sudoku.in
0 0 0 7 0 2 4 5 3
9 0 0 0 0 8 0 0 0
7 4 0 0 0 5 0 1 0
1 9 5 0 8 0 0 0 0
0 7 0 0 0 0 0 2 5
0 3 0 5 7 9 1 0 8
0 0 0 6 0 1 0 0 0
0 6 0 9 0 0 0 0 1
0 0 0 0 0 0 0 0 6
sudoku.out
2852
【数据范围】
40%的数据,数独中非 0数的个数不少于 30。
80%的数据,数独中非 0数的个数不少于 26。
100%的数据,数独中非 0 数的个数不少于 24。
SOLUTION:
没有什么说的直接大暴搜吧,只要考虑了三种限制条件每行每列每一小块(这根本不算剪枝),就可以过了。
1 #include<cstdio> 2 #include<cstring> 3 #include<iostream> 4 #include<algorithm> 5 using namespace std; 6 int a[10][10],score[10][10],ans,tot,ji[100],belong[10][10]; 7 bool hang[10][10],lie[10][10],blo[10][10]; 8 bool check() { 9 for(int i=1; i<=9; i++) 10 for(int j=1; j<=9; j++) 11 if(!hang[i][j]||!lie[i][j])return false; 12 return true; 13 } 14 inline void dfs(int i,int j,int sum) { 15 if(i==1&&j==1) { 16 if(a[i][j]) {ans=max(ans,sum);return ;} 17 int num; 18 for(num=1; num<=9; num++) { 19 if((hang[1][num]&&!lie[1][num])||(!hang[1][num]&&lie[1][num])) return ; 20 if(!hang[1][num]&&!lie[1][num]) break; 21 }ans=max(ans,sum+score[1][1]*num); 22 return ; 23 } 24 if(a[i][j]) { 25 if(i&1) 26 if(j>1) dfs(i,j-1,sum); 27 else dfs(i-1,j,sum); 28 else 29 if(j<9)dfs(i,j+1,sum); 30 else dfs(i-1,j,sum); 31 return ; 32 } 33 for(int num=1; num<=9; num++) { 34 if(hang[i][num]||lie[j][num]||blo[belong[i][j]][num]) continue; 35 hang[i][num]=1; lie[j][num]=1; 36 a[i][j]=num;blo[belong[i][j]][num]=1; 37 sum+=score[i][j]*num; 38 if(i&1) 39 if(j>1) dfs(i,j-1,sum); 40 else dfs(i-1,j,sum); 41 else 42 if(j<9) dfs(i,j+1,sum); 43 else dfs(i-1,j,sum); 44 hang[i][num]=0; lie[j][num]=0; 45 a[i][j]=0; blo[belong[i][j]][num]=0;sum-=score[i][j]*num; 46 } 47 return ; 48 } 49 void init() { 50 for(int i=1; i<=3; i++) { 51 for(int j=1; j<=3; j++) belong[i][j]=1; 52 for(int j=4; j<=6; j++) belong[i][j]=2; 53 for(int j=7; j<=9; j++) belong[i][j]=3; 54 } 55 for(int i=4; i<=6; i++) { 56 for(int j=1; j<=3; j++) belong[i][j]=4; 57 for(int j=4; j<=6; j++) belong[i][j]=5; 58 for(int j=7; j<=9; j++) belong[i][j]=6; 59 } 60 for(int i=7; i<=9; i++) { 61 for(int j=1; j<=3; j++) belong[i][j]=7; 62 for(int j=4; j<=6; j++) belong[i][j]=8; 63 for(int j=7; j<=9; j++) belong[i][j]=9; 64 } 65 score[5][5]=10;score[4][4]=score[6][6]=score[4][6]=score[6][4]=score[4][5]=score[5][4]=score[5][6]=score[6][5]=9; 66 for(int j=3; j<=7; j++) score[3][j]=score[7][j]=8; 67 for(int i=4; i<=6; i++) score[i][3]=score[i][7]=8; 68 for(int j=2; j<=8; j++)score[2][j]=score[8][j]=7; 69 for(int i=3; i<=7; i++) score[i][2]=score[i][8]=7; 70 for(int j=1; j<=9; j++) score[1][j]=score[9][j]=6; 71 for(int i=2; i<=8; i++) score[i][1]=score[i][9]=6; 72 } 73 int main() { 74 freopen("sudoku.in","r",stdin); 75 freopen("sudoku.out","w",stdout); 76 init(); 77 int res=0; 78 for(int i=1; i<=9; i++) { 79 for(int j=1; j<=9; j++) { 80 scanf("%d",&a[i][j]); 81 if(!a[i][j]) 82 continue; 83 res+=a[i][j]*score[i][j]; 84 hang[i][a[i][j]]=1; 85 lie[j][a[i][j]]=1; 86 blo[belong[i][j]][a[i][j]]=1; 87 } 88 }dfs(9,9,res); 89 if(!ans) cout<<"-1"; 90 else cout<<ans; 91 return 0; 92 }
以上是关于[NOIP2009]靶形数独 题解的主要内容,如果未能解决你的问题,请参考以下文章