NOIP2009靶形数独(暴搜)

时间:2021-10-13 18:28:38

题目传送门


题目描述

小城和小华都是热爱数学的好学生,最近,他们不约而同地迷上了数独游戏,好胜的他们想用数独来一比高低。但普通的数独对他们来说都过于简单了,于是他们向Z博士请教,Z博士拿出了他最近发明的「靶形数独」,作为这两个孩子比试的题目。

靶形数独的方格同普通数独一样,在9格宽×9格高 的大九宫格中有9个3格宽×3格高 的小九宫格(用粗黑色线隔开的)。在这个大九宫格中,有一些数字是已知的,根据这些数字,利用逻辑推理,在其他的空格上填入1到9的数字。每个数字在每个小九宫格内不能 重复出现,每个数字在每行、每列也不能重复出现。但靶形数独有一点和普通数独不同,即 每一个方格都有一个分值,而且如同一个靶子一样,离中心越近则分值越高。(如图)

NOIP2009靶形数独(暴搜)

上图具体的分值分布是:里面一格(黄色区域)为10分,黄色区域外面的一圈(红色区域)每个格子为9分,再外面一圈(蓝色区域)每个格子为8分,蓝色区域外面一圈(棕色区域)每个格子为7分,外面一圈(白色区域)每个格子为6分,如上图所示。

比赛的要求是:每个人必须完成一个给定的数独(每个给定数独可能有不同的填法),而且要争取更高的总分数。而这个总分数即每个方格上的分值和完成这个数独时填在相应格上的数字的乘积的总和。

如图,在以下的这个已经填完数字的靶形数独游戏中,总分数为2829。游戏规定,将以总分数的高低决出胜负。

NOIP2009靶形数独(暴搜)

由于求胜心切,小城找到了善于编程的你,让你帮他求出,对于给定的靶形数独,能够得到的高分数。


输入格式

输入一共9行。每行9个整数(每个数都在0-9的范围内),表示一个尚未填满的数独方格,未填的空格用0表示。每两个数字之间用一个空格隔开。


输出格式

输出共1行。
输出可以得到的靶形数独的高分数。如果这个数独无解,则输出整数-1 。


样例

样例输入1:

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

样例输出1:

2829

样例输入2:

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

样例输出2:

2852


数据范围与提示

40%的数据,数独中非0数的个数不少于30。
80%的数据,数独中非0数的个数不少于26。
100%的数据,数独中非0数的个数不少于24。


题解

一看就是搜索,分分中打出爆搜代码。

O(${10}^{81}$)

淡定……

剪枝,不行的情况直接剪掉。

等等,TLE 80?!?!

震惊!!!

状压,没错~

懒得打,也没错~

既然他会出现在卡长的专题

考虑到出题人在出数据的时候一般都是正搜卡掉你,那么倒着搜。

再加点调料,register……

TLE 95,本宝宝瞬间就不喜欢你了,555……

居然倒着搜还卡我,我这么可爱>_<……

那我,从左下往右上?

Judging……

Accepted!!!

哇~

瞬间给你一个温暖的拥抱~


代码时刻

#include<bits/stdc++.h>
using namespace std;
int ans,rec;
int Map[30][30];
int wzc[10][10]={0,0,0,0,0,0,0,0,0,0,0,6,6,6,6,6,6,6,6,6,0,6,7,7,7,7,7,7,7,6,0,6,7,8,8,8,8,8,7,6,0,6,7,8,9,9,9,8,7,6,0,6,7,8,9,10,9,8,7,6,0,6,7,8,9,9,9,8,7,6,0,6,7,8,8,8,8,8,7,6,0,6,7,7,7,7,7,7,7,6,0,6,6,6,6,6,6,6,6,6};
int miao[10][10]={0,0,0,0,0,0,0,0,0,0,0,1,1,1,2,2,2,3,3,3,0,1,1,1,2,2,2,3,3,3,0,1,1,1,2,2,2,3,3,3,0,4,4,4,5,5,5,6,6,6,0,4,4,4,5,5,5,6,6,6,0,4,4,4,5,5,5,6,6,6,0,7,7,7,8,8,8,9,9,9,0,7,7,7,8,8,8,9,9,9,0,7,7,7,8,8,8,9,9,9};
int h[10][10],l[10][10],q[10][10];//预处理两张表
void _doudou(register int x,register int y,register int sor)//从左下往右上搜~
{
if(y==10){_doudou(x-1,1,sor);return;}
if(!x){ans=max(ans,sor);return;}
if(Map[x][y]){_doudou(x,y+1,sor);return;}
for(register int i=9;i;i--)
if(!h[x][i]&&!l[y][i]&&!q[miao[x][y]][i])
{
h[x][i]^=1,l[y][i]^=1,q[miao[x][y]][i]^=1;//异或跑得快~
_doudou(x,y+1,sor+i*wzc[x][y]);
h[x][i]^=1,l[y][i]^=1,q[miao[x][y]][i]^=1;
}
}
int main()
{
for(register int i=1;i<=9;i++)
for(register int j=1;j<=9;j++)
{
scanf("%d",&Map[i][j]);
rec+=Map[i][j]*wzc[i][j];
h[i][Map[i][j]]^=1;
l[j][Map[i][j]]^=1;
q[miao[i][j]][Map[i][j]]^=1;
}
_doudou(9,1,rec);
if(!ans)printf("-1");
else printf("%d",ans);
return 0;
}

rp++