[NOIP2009]靶形数独 题解

2024-03-24 17:30
文章标签 题解 形数 noip2009

本文主要是介绍[NOIP2009]靶形数独 题解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

407. [NOIP2009] 靶形数独

时间限制: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。  

  这道题在这里先藐视一下有160的身高却又190的心的jhs,我第一次提交剪枝少了一些,被他嘲笑,结果A了之后还比他快了1/3,(自带元首渣渣配音)。

  这道题其实考的是深搜和剪枝,首先,本题是要求最大值,因此以往当前ans>最小(优)ans的剪枝报废。然后我们可以思考一下,我们并没有必要去时常检验数独,而是直接去通过bool数组去限制填数,因为搜索题有一个定律,再好的剪枝也不如在搜索时直接不去对该情况进行搜索。

  因此我们通过3个bool数组对三种情况进行限制,就会很开心的发现自己T了一个点,这时候怎么办呢?我们可以反其道而行之,倒着搜,从某中意义上来讲,如果一个题目它无意去卡你这点,他们一般都会在dfs搜到较浅层的时候就发展出很多枝;来考验大家的剪枝能力,大家可以这样去想,我们可以把搜索看成一个头重脚轻的类似陀螺形状的东西,倒着搜索是要快很多的。

 1 #include<iostream>
 2 #include<cstdlib>
 3 #include<cstdio>
 4 #include<cstring>
 5 #include<algorithm>
 6 #include<map>
 7 #include<queue>
 8 #include<string>
 9 #include<cmath>
10 using namespace std;
11 int n=9,a[15][15],ans;
12 bool fw[2][15][15],be[15][15];
13 int f[15][15],num[15][15];
14 int get(int x,int y){
15     if(x<=3&&y<=3) return 1;
16     if(x<=3&&y<=6) return 2;
17     if(x<=3&&y<=9) return 3;
18     if(x<=6&&y<=3) return 4;
19     if(x<=6&&y<=6) return 5;
20     if(x<=6&&y<=9) return 6;
21     if(x<=9&&y<=3) return 7;
22     if(x<=9&&y<=6) return 8;
23     return 9;
24 }
25 void dfs(int l,int wz,int sum){
26     if(wz==n+1)
27     {
28         if(l==1)
29         {
30             if(sum>ans) ans=sum;
31             return;
32         }
33         dfs(l-1,0,sum);
34         return;
35     }
36     int bj=0;
37     for(int i=wz+1;i<=n;i++)
38     {
39         if(!a[l][i])
40         {
41             bj=i;
42             break;
43         }
44     }
45     if(!bj)
46     {
47         dfs(l,n+1,sum);
48         return;
49     }
50     for(int i=1;i<=9;i++)
51     {
52         if(!fw[0][l][i]&&!fw[1][bj][i]&&!be[num[l][bj]][i])
53         {
54             be[num[l][bj]][i]=fw[0][l][i]=fw[1][bj][i]=1;
55             dfs(l,bj,sum+i*f[l][bj]);
56             be[num[l][bj]][i]=fw[0][l][i]=fw[1][bj][i]=0;
57         }
58     }
59 }
60 int main(){
61     int sum=0;
62     for(int i=1;i<=n;i++)
63         f[1][i]=f[i][1]=f[n][i]=f[i][n]=6;
64     for(int i=2;i<=n-1;i++)
65         f[2][i]=f[i][2]=f[n-1][i]=f[i][n-1]=7;
66     for(int i=3;i<=n-2;i++)
67         f[3][i]=f[i][3]=f[n-2][i]=f[i][n-2]=8;
68     for(int i=4;i<=n-3;i++)
69         f[4][i]=f[i][4]=f[n-3][i]=f[i][n-3]=9;
70     f[5][5]=10;
71     for(int i=1;i<=n;i++)
72     {
73         for(int j=1;j<=n;j++)
74         {
75             scanf("%d",&a[i][j]);
76             fw[0][i][a[i][j]]=1;
77             fw[1][j][a[i][j]]=1;
78             num[i][j]=get(i,j);
79             be[num[i][j]][a[i][j]]=1;
80             sum+=a[i][j]*f[i][j];
81         }
82     }
83      
84     dfs(n,0,sum);
85     if(!ans) ans=-1;
86     printf("%d\n",ans);
87 //  while(1);
88     return 0;
89 }
View Code

 

转载于:https://www.cnblogs.com/liutianrui/p/7327751.html

这篇关于[NOIP2009]靶形数独 题解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/842260

相关文章

C++ | Leetcode C++题解之第393题UTF-8编码验证

题目: 题解: class Solution {public:static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num &

C语言 | Leetcode C语言题解之第393题UTF-8编码验证

题目: 题解: static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num & MASK1) == 0) {return

C - Word Ladder题解

C - Word Ladder 题解 解题思路: 先输入两个字符串S 和t 然后在S和T中寻找有多少个字符不同的个数(也就是需要变换多少次) 开始替换时: tips: 字符串下标以0开始 我们定义两个变量a和b,用于记录当前遍历到的字符 首先是判断:如果这时a已经==b了,那么就跳过,不用管; 如果a大于b的话:那么我们就让s中的第i项替换成b,接着就直接输出S就行了。 这样

【秋招笔试】9.07米哈游秋招改编题-三语言题解

🍭 大家好这里是 春秋招笔试突围,一起备战大厂笔试 💻 ACM金牌团队🏅️ | 多次AK大厂笔试 | 大厂实习经历 ✨ 本系列打算持续跟新 春秋招笔试题 👏 感谢大家的订阅➕ 和 喜欢💗 和 手里的小花花🌸 ✨ 笔试合集传送们 -> 🧷春秋招笔试合集 🍒 本专栏已收集 100+ 套笔试题,笔试真题 会在第一时间跟新 🍄 题面描述等均已改编,如果和你笔试题看到的题面描述

LeetCode 第414场周赛个人题解

目录 Q1. 将日期转换为二进制表示 原题链接 思路分析 AC代码 Q2. 范围内整数的最大得分 原题链接 思路分析 AC代码 Q3. 到达数组末尾的最大得分 原题链接 思路分析 AC代码 Q4. 吃掉所有兵需要的最多移动次数 原题链接 思路分析 AC代码 Q1. 将日期转换为二进制表示 原题链接 Q1. 将日期转换为二进制表示 思路分析

牛客小白月赛100部分题解

比赛地址:牛客小白月赛100_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ A.ACM中的A题 #include<bits/stdc++.h>using namespace std;#define ll long long#define ull = unsigned long longvoid solve() {ll a,b,c;cin>>a>>b>

P2858 [USACO06FEB] Treats for the Cows G/S 题解

P2858 题意 给一个数组。每天把最左或者最右的东西卖掉,第 i i i个东西,第 d a y day day天卖出的价格是 a [ i ] ∗ d a y a[i]*day a[i]∗day。 记忆化搜索 void dfs(int l,int r,int day,ll sum){if(v[l][r]>=sum)return;v[l][r]=sum;if(l>r)//这就是dp答案{

【C++题解】1272. 郭远摘苹果

欢迎关注本专栏《C++从零基础到信奥赛入门级(CSP-J)》 问题:1272. 郭远摘苹果 类型:二维数组 题目描述: 郭远有一天走到了一片苹果林,里面每颗树上都结有不同数目的苹果,郭远身上只能拿同一棵树上的苹果,他每到一棵果树前都会把自己身上的苹果扔掉并摘下他所在树上的苹果并带走(假设郭远会走过每一棵苹果树),问在郭远摘苹果的整个过程中,他身上携带的最多苹果数与最小苹果数的差是多少?

【最新华为OD机试E卷-支持在线评测】机器人活动区域(100分)多语言题解-(Python/C/JavaScript/Java/Cpp)

🍭 大家好这里是春秋招笔试突围 ,一枚热爱算法的程序员 ✨ 本系列打算持续跟新华为OD-E/D卷的三语言AC题解 💻 ACM金牌🏅️团队| 多次AK大厂笔试 | 编程一对一辅导 👏 感谢大家的订阅➕ 和 喜欢💗 🍿 最新华为OD机试D卷目录,全、新、准,题目覆盖率达 95% 以上,支持题目在线评测,专栏文章质量平均 94 分 最新华为OD机试目录: https://blog.

2023 CCPC(秦皇岛)现场(第二届环球杯.第 2 阶段:秦皇岛)部分题解

所有题目链接:Dashboard - The 2023 CCPC (Qinhuangdao) Onsite (The 2nd Universal Cup. Stage 9: Qinhuangdao) - Codeforces 中文题面: contest-37054-zh.pdf (codeforces.com) G. Path 链接: Problem - G - Codeforces