LuoguP1074[NOIP2009] 靶形数独 解题报告【搜索】

来源:互联网 发布:适合抓握的鼠标 知乎 编辑:程序博客网 时间:2024/06/05 08:37

题目描述
小城和小华都是热爱数学的好学生,最近,他们不约而同地迷上了数独游戏,好胜的他们想用数独来一比高低。但普通的数独对他们来说都过于简单了,于是他们向 Z 博士请教,Z 博士拿出了他最近发明的“靶形数独”,作为这两个孩子比试的题目。
靶形数独的方格同普通数独一样,在 9 格宽×9 格高的大九宫格中有 9 个 3 格宽×3 格高的小九宫格(用粗黑色线隔开的)。在这个大九宫格中,有一些数字是已知的,根据这些数字,利用逻辑推理,在其他的空格上填入 1 到 9 的数字。每个数字在每个小九宫格内不能重复出现,每个数字在每行、每列也不能重复出现。但靶形数独有一点和普通数独不同,即每一个方格都有一个分值,而且如同一个靶子一样,离中心越近则分值越高。(如图)

上图具体的分值分布是:最里面一格(黄色区域)为 10 分,黄色区域外面的一圈(红色区域)每个格子为 9 分,再外面一圈(蓝色区域)每个格子为 8 分,蓝色区域外面一圈(棕色区域)每个格子为 7 分,最外面一圈(白色区域)每个格子为 6 分,如上图所示。比赛的要求是:每个人必须完成一个给定的数独(每个给定数独可能有不同的填法),而且要争取更高的总分数。而这个总分数即每个方格上的分值和完成这个数独时填在相应格上的数字的乘积的总和总分数即每个方格上的分值和完成这个数独时填在相应格上的数字的乘积的总和。如图,在以下的这个已经填完数字的靶形数独游戏中,总分数为 2829。游戏规定,将以总分数的高低决出胜负。

由于求胜心切,小城找到了善于编程的你,让你帮他求出,对于给定的靶形数独,能够得到的最高分数。
输入输出格式
输入格式:
一共 9 行。每行 9 个整数(每个数都在 0―9 的范围内),表示一个尚未填满的数独方格,未填的空格用“0”表示。每两个数字之间用一个空格隔开。
输出格式:
输出文件 sudoku.out 共 1 行。
输出可以得到的靶形数独的最高分数。如果这个数独无解,则输出整数-1。
输入输出样例
输入样例#1:
sudoku1
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
sudoku2
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
输出样例#1:
sudoku1
2829
sudoku2
2852
说明
【数据范围】
40%的数据,数独中非 0 数的个数不少于 30。
80%的数据,数独中非 0 数的个数不少于 26。
100%的数据,数独中非 0 数的个数不少于 24。
NOIP 2009 提高组 第四题
解题报告
不难看出,由于题目的限制,即每行、每列、每宫格的数字不能超过9个且为1~9,再加上给了我们一张填好的数独,所以一定有一些格子可以填的数目相对较少,这些格子当然要考虑优先去填。具体的讲,我们用二维数组记录每一行、每一列分别有几个数被填充了,找到最大的行和最大的列(理论上还要找最大的宫格),从这一格开始搜索,过后在搜索耳朵转移过程中也是如法炮制。
代码如下:

#include<cstdio>#include<cstring>#include<algorithm>#define N 9using namespace std;int w[N+5][N+5]={    {0,0,0,0,0,0,0,0,0,0,0},    {0,6,6,6,6,6,6,6,6,6,0},    {0,6,7,7,7,7,7,7,7,6,0},    {0,6,7,8,8,8,8,8,7,6,0},    {0,6,7,8,9,9,9,8,7,6,0},    {0,6,7,8,9,10,9,8,7,6,0},    {0,6,7,8,9,9,9,8,7,6,0},    {0,6,7,8,8,8,8,8,7,6,0},    {0,6,7,7,7,7,7,7,7,6,0},    {0,6,6,6,6,6,6,6,6,6,0},    {0,0,0,0,0,0,0,0,0,0,0}};int ans=-1,cnt;int map[N+5][N+5],flag_r[N+5][N+5],flag_c[N+5][N+5],flag_pos[N+5][N+5];//flag_r[i][num]记录第i行num这个数有没有出现int cnt_r[N+5],cnt_c[N+5];//cnt_r[i]记录第i行填充了几个数字int id[N+5][N+5];inline int read(){    int x=0,f=1;    register char ch=getchar();    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}    while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();    return x*f;}inline int get_id(int i,int j){    return id[i][j]!=0?id[i][j]:id[i][j]=(i-1)/3*3+1+(j-1)/3;}inline int cal(){    int ret=0;    for(int i=1;i<=N;i++)    for(int j=1;j<=N;j++)ret+=w[i][j]*map[i][j];    return ret;}void dfs(int r,int c,int tot){    if(tot==N*N)    {        ans=max(ans,cal());        return ;    }    for(int num=1;num<=N;num++)    {        if(flag_r[r][num]||flag_c[c][num]||flag_pos[get_id(r,c)][num]) continue;        flag_r[r][num]=1,flag_c[c][num]=1,flag_pos[get_id(r,c)][num]=1;        cnt_r[r]++,cnt_c[c]++;        map[r][c]=num;        int vmax_r=-1,vmax_c=-1,r1=0,c1=0;        for(int i=1;i<=N;i++)if(cnt_r[i]>vmax_r&&cnt_r[i]<9)vmax_r=cnt_r[i],r1=i;        for(int j=1;j<=N;j++)if(cnt_c[j]>vmax_c&&(!map[r1][j]))vmax_c=cnt_c[j],c1=j;        dfs(r1,c1,tot+1);        flag_r[r][num]=0,flag_c[c][num]=0,flag_pos[get_id(r,c)][num]=0;        cnt_r[r]--,cnt_c[c]--;        map[r][c]=0;    }}int main(){    for(int i=1;i<=N;i++)    for(int j=1;j<=N;j++)    {        map[i][j]=read();        if(map[i][j]!=0)        {            flag_r[i][map[i][j]]=1,flag_c[j][map[i][j]]=1,flag_pos[get_id(i,j)][map[i][j]]=1;            cnt_r[i]++,cnt_c[j]++;            cnt++;        }    }    int vmax_r=-1,vmax_c=-1,r,c;    for(int i=1;i<=N;i++)if(cnt_r[i]>vmax_r&&cnt_r[i]<9)vmax_r=cnt_r[i],r=i;    for(int j=1;j<=N;j++)if(cnt_c[j]>vmax_c&&(!map[r][j]))vmax_c=cnt_c[j],c=j;    dfs(r,c,cnt);    printf("%d",ans);    return 0;}
原创粉丝点击