蓝桥杯第100 题 九宫幻方 DFS 全排列 C++ 解题思维

news2024/12/23 9:23:02

题目

九宫幻方icon-default.png?t=N7T8https://www.lanqiao.cn/problems/100/learning/?page=1&first_category_id=1&name=%E4%B9%9D

思路和解题方法  一 (DFS)

  1. 首先,定义了一些全局变量和数组。vis数组用于标记已经出现过的数字,a数组用于存储数独的初始状态和中间状态,ans数组用于存储找到的解决方案,p数组用于存储空白格子的坐标,n表示空白格子的数量,cnt记录解决方案的数量。

  2. check()函数用于检查当前填充的数字是否满足数独的规则。它首先计算对角线上的元素之和,然后检查每行和每列的和是否相等,并返回一个布尔值表示结果。

  3. dfs()函数是核心部分,使用递归实现深度优先搜索。它从第一个空白格子开始填充数字,每次填充一个数字后,递归调用自身继续填充下一个空白格子,直到所有空白格子都填充完毕。如果找到了符合数独规则的解决方案,则将其记录在ans数组中,并增加cnt的计数。

  4. main()函数中,首先读入数独的初始状态,并初始化相关变量。然后调用dfs()函数进行搜索。最后,根据搜索结果输出解决方案或提示"Too Many"表示找到了多个解决方案。

需要注意的是,该程序假设输入的数独有唯一解或没有解。如果存在多个解,则只会输出其中一个解。如果不存在解,则输出"Too Many"表示无法确定唯一解。

c++ 代码  1

#include<bits/stdc++.h>
using namespace std;

int vis[10],a[5][5],ans[5][5];
// vis数组用来标记数字是否使用过,a数组记录数独题目的矩阵,ans数组用来存储解决方案
int n=1,cnt;
pair<int ,int>p[10];
// p数组用来存储空白格的坐标,n记录空白格的个数
bool check(){
    int sum=a[1][1]+a[2][2]+a[3][3];
    // 先计算左上到右下的斜线上的数字之和
    if(sum!=a[1][3]+a[2][2]+a[3][1]) return false;
    // 再计算右上到左下的斜线上的数字之和,如果不相等,则不是数独的解
    for(int i=1;i<=3;i++){
    int tmp1=0,tmp2=0;
    for(int j=1;j<=3;j++)tmp1+=a[i][j],tmp2+=a[j][i];//行和列的和相等 
        if(tmp1!=sum||tmp2!=sum) return false;
    }
    // 分别计算每行每列的数字之和,如果与斜线上的数字之和不相等,则不是数独的解
    return true;
} 

void dfs(int now){
    if(now>=n){
        if(check()){
            cnt++;
            for(int i=1;i<=3;i++)
                for(int j=1;j<=3;j++)
                    ans[i][j]=a[i][j];
            // 如果当前数独矩阵是解,则令ans数组等于当前矩阵
        }
        return;
    }
    int x=p[now].first,y=p[now].second;
    for(int k=1;k<=9;k++)
    {
        if(vis[k])continue;
        a[x][y]=k;
        vis[k]=1;
        dfs(now+1);
        a[x][y]=0;
        vis[k]=0;
        // 搜索之前先标记数字已经使用过,搜索完之后再回溯
    }
}

int main()
{
    for(int i=1;i<=3;i++)
        for(int j=1;j<=3;j++){
            cin>>a[i][j];
            if(!a[i][j])p[n++]=make_pair(i,j);
            // 记录空白格的坐标
            vis[a[i][j]]=1;    
            // 标记数字已经使用过
        }
    dfs(1);//从第一个空白格开始搜索
    if(cnt==1){
        for(int i=1;i<=3;i++)
            for(int j=1;j<=3;j++)
                cout<<ans[i][j]<<" \n"[j==3];
            }
    else cout<<"Too Many\n";
    // 如果有多个解,则输出"Too Many"
    return 0;
 }

 思路和解题方法   二 (全排列)

  1. 从标准输入中读入一个包含9个整数的数组a。
  2. 使用next_permutation函数生成arr数组的全排列,并逐个判断是否满足数独的条件。
  3. 如果某个排列满足数独条件,并且与输入数组a匹配,则将该排列赋值给数组a,并增加计数器ans的值。
  4. 最后根据计数器ans的值输出结果。如果有且仅有一个解,则输出解;如果有多个解,则输出"Too Many"。

c++ 代码

#include<iostream>
#include<algorithm>
#include<bits/stdc++.h> //包含一些常用的头文件,例如vector、string等
using namespace std;

int a[10]; //定义一个长度为10的数组a,用于存储输入的数独初始状态
int ans = 0; //计数器,用于存储符合数独条件的解的个数
int arr[10] = {1,2,3,4,5,6,7,8,9}; //定义一个长度为10的数组arr,用于生成全排列

//检查当前排列是否符合数独条件
bool check(int a[])
{
    int a1=a[0]+a[1]+a[2];
    int a2=a[3]+a[4]+a[5];
    int a3=a[6]+a[7]+a[8];
    int a4=a[0]+a[3]+a[6];
    int a5=a[1]+a[4]+a[7];
    int a6=a[2]+a[5]+a[8];
    int a7=a[2]+a[4]+a[6];
    int a8=a[0]+a[4]+a[8];

    if(a1==a2&&a2==a3&&a3==a4&&a4==a5&&a5==a6&&a6==a7&&a7==a8) //如果8个和相等,则返回true
    {
        return true;
    }
    return false; //否则返回false
}

int main()
{
    for(int i = 0;i<9;i++) //读入初始数独状态
    {
        cin>>a[i];
    }

    while(next_permutation(arr,arr+9)){ //生成arr数组的全排列,逐个检查是否符合数独条件
        if(check(arr)) //如果符合数独条件
        {
            bool y = true;
            for(int i = 0;i<9;i++)
            {
                if(a[i]!=0&&a[i]!=arr[i]) //如果输入的数独状态不为0且与当前排列不一致,则说明不是想要的解
                {
                    y = false; //设置标志y为false
                }
            }
            if(y) //如果当前排列与输入的数独状态匹配
            {
                for(int i = 0;i<9;i++) //将当前排列赋值给数组a
                {
                    a[i] = arr[i];
                }
                ans++; //计数器ans加1
            }
        }
    }

    if(ans == 1) //如果只有一个解,则输出该解
    {
        int cnt = 0; //计数器cnt,用于输出格式控制
        for(int i = 0;i<9;i++)
        {
            if(cnt==3)
            {
                cout<<endl; //每输出三个数字换行
                cnt = 0;
            }
            cout<<a[i]<<" "; //输出数独解
            cnt++; //计数器加1
        }
    }else { //如果有多个解,则输出"Too Many"
        printf("Too Many");
    }
    return 0;
}

知识点解释

next_permutation() 全排列

next_permutation()是一个函数,通常在编程语言中用于生成给定序列的下一个排列。它可以按照字典序(升序)生成给定序列的下一个排列,并将其更新为下一个排列。

具体而言,next_permutation()函数接受一个序列作为参数,并将该序列重排为下一个字典序更大的排列。如果没有下一个更大的排列,则将序列重排为最小的(升序)排列。该函数返回一个布尔值,指示是否成功生成了下一个排列。

下面是一个示例,展示了如何使用next_permutation()函数来生成给定序列的所有排列:

#include <iostream>
 #include <algorithm>
 using namespace std;
 ​
 int main() {
     int arr[] = {1, 2, 3};
 ​
     // 生成并打印所有排列
     do {
         for (int i = 0; i < 3; i++) {
             cout << arr[i] << " ";
         }
         cout << endl;
     } while (next_permutation(arr, arr + 3));
 ​
     return 0;
 }

输出结果将是:

 1 2 3
 1 3 2
 2 1 3
 2 3 1
 3 1 2
 3 2 1

这个例子展示了如何使用next_permutation()函数生成给定序列的所有排列,并将它们打印出来。每次调用next_permutation()函数时,原始数组会被修改为下一个排列,直到没有更多的排列可生成为止。

需要注意的是,next_permutation()函数在不同的编程语言和库中可能有略微不同的实现方式,但其基本功能是相似的:生成给定序列的下一个排列。

觉得有用的话可以点点赞,支持一下。

如果愿意的话关注一下。会对你有更多的帮助。

每天都会不定时更新哦  >人<  。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1261207.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

A start job is running for Hold unt…s up (1d 18h 52min 25s / no limit) 如何去掉

在host串口里一直出现打印 A start job is running for Hold unt…s up (1d 18h 52min 25s / no limit) 这个是有一个进程一直在执行中&#xff0c;那么是什么呢&#xff1f;因为我的host通过SSH连接后就可以进入host shell界面了。那这个线程是什么程序导致的呢&#xff1f; …

Windows全系列 本地密码暴力破解

首先 咱们要准备两个工具&#xff1a; 第一个是 pwdump-master 第二个是 saminside_softradar-com.exe这两个工具 我会一并上传 需要的同学 可以自取本文章操作思路是&#xff1a; 第一步 首先把我刚刚提到的两个软件 以某种手段放置于机器中 如果是真实机 就用U盘 拷贝到真实机…

力扣 hot100 最小覆盖子串 滑动窗口 字符计数

&#x1f468;‍&#x1f3eb; 题目地址 &#x1f37b; AC code class Solution {public String minWindow(String s, String t){int n s.length();int m t.length();if (n < m)return "";char[] tt t.toCharArray();int[] cnt new int[128];// 字符计数数组…

NOIP2007提高组第二轮T3:矩阵取数游戏

题目链接 [NOIP2007 提高组] 矩阵取数游戏 题目描述 帅帅经常跟同学玩一个矩阵取数游戏&#xff1a;对于一个给定的 n m n \times m nm 的矩阵&#xff0c;矩阵中的每个元素 a i , j a_{i,j} ai,j​ 均为非负整数。游戏规则如下&#xff1a; 每次取数时须从每行各取走一…

【扩散模型】DDIM从原理到实战

DDIM从原理到实战 1. DDIM简介2. 实战DDIM2.1 载入一个预训练过的pipeline2.2 DDIM采样2.3 反转&#xff08;invert&#xff09; 3. 组合封装参考资料 DDPM过程的一个问题是在训练后生成图像的速度。当然&#xff0c;我们可能能够生成令人惊叹的图像&#xff0c;但生成一张图像…

【算法】NOIP2003神经网络

题目描述 人工神经网络&#xff08;Artificial Neural Network&#xff09;是一种新兴的具有自我学习能力的计算系统&#xff0c;在模式识别、函数逼近及贷款风险评估等诸多领域有广泛的应用。对神经网络的研究一直是当今的热门方向&#xff0c;兰兰同学在自学了一本神经网络的…

pycharm安装PyQt5及其工具

PyCharm安装PyQt5及其工具&#xff08;Qt Designer、PyUIC、PyRcc&#xff09;详细教程_pycharm pyqt5-CSDN博客 上面是原文链接&#xff0c;根据原文链接&#xff0c;我重新记录一下。IDE&#xff1a;pycharm 2023.2.5 一共需要安装5个。 在PyCharm中如何完整优雅地安装配置…

ESP32-Web-Server编程- JS 基础 3

ESP32-Web-Server编程- JS 基础 3 概述 本示例演示通过 button 控件的 onclick 内联属性&#xff0c;实现在网页上点击按钮&#xff0c;切换 LED 灯图标的转变。 示例解析 前端设计 前端代码建立了一个 id 为 “imageLamp” 的图片对象。并建立两个按钮&#xff0c;设计两…

【WP】Geek Challenge 2023 web 部分wp

EzHttp http协议基础题 unsign 简单反序列化题 n00b_Upload 很简单的文件上传&#xff0c;上传1.php&#xff0c;抓包&#xff0c;发现php内容被过滤了&#xff0c;改为<? eval($_POST[‘a’]);?>&#xff0c;上传成功&#xff0c;命令执行读取就好了 easy_php …

企业软件手机app定制开发趋势|小程序网站搭建

企业软件手机app定制开发趋势|小程序网站搭建 随着移动互联网的快速发展和企业数字化转型的加速&#xff0c;企业软件手机App定制开发正成为一个新的趋势。这种趋势主要是由于企业对于手机App的需求增长以及现有的通用应用不能满足企业特定需求的情况下而产生的。 1.企业软件手…

ensp 启动设备时报40错误,然后一直没有去管,再次进去就好了,我知道是配置虚拟机的时候修改了一些设置:

第一个阶段&#xff1a; 那时我是重置电脑之后就安装了ensp所以没有出现什么问题&#xff0c;&#xff08;那时没有导入ce6800和12800还有防火墙6000&#xff09; 第二个阶段&#xff1a; 因为有华为相关的实验要做&#xff0c;所以心血来潮打开了ensp&#xff08;路由器之前…

Maven Helper插件——实现一键Maven依赖冲突问题

总结/朱季谦 业余在一个SpringBoot项目集成Swagger2时&#xff0c;启动过程一直出现以下报错信息—— An attempt was made to call a method that does not exist. The attempt was made from the following location: ​ springfox.documentation.schema.DefaultModelDepe…

Redis面试题:分片集群相关问题

目录 面试官&#xff1a;redis的分片集群有什么作用 面试官&#xff1a;Redis分片集群中数据是怎么存储和读取的&#xff1f; 面试官&#xff1a;redis的分片集群有什么作用 候选人&#xff1a;分片集群主要解决的是&#xff0c;海量数据存储的问题&#xff0c;集群中有多个m…

Selenium 学习(0.16)——软件测试之测试用例设计方法——白盒测试——逻辑覆盖法(语句覆盖和判定覆盖)

写在前面 今天回来有点晚&#xff0c;因为上午给小伙伴们开了个小会&#xff0c;随便说了些什么&#xff0c;结果小伙伴们下班就没急着走&#xff0c;弄点我还有点不好意思提前走了&#xff0c;就略留了一会。其实也没说什么&#xff0c;就是强调工作要抓点紧&#xff0c;8小时…

FLASK博客系列7——我要插入数据库

我们来继续上次的内容&#xff0c;实现将数据插入数据库。 我们先更改下models.py&#xff0c;由于上次笔误&#xff0c;把外键关联写错了。在这里给大家说声抱歉。不过竟然没有小伙伴发现。 models.py from app import dbclass User(db.Model): # 表名将会是 user&#xff0…

GPIO的使用--操作PE02 PE03 PE04实现开关控制灯泡亮灭

效果&#xff1a; 开关控制灯的亮灭 目录 1.找到引脚组别(DEFG) led灯硬件结构 开关硬件结构 2.时钟使能 3.GPIO时钟控制 4.控制实现思路 5. 完整代码 6.视频演示 1.找到引脚组别(DEFG) 开关的引脚组别--E&#xff1b;LED灯的引脚组别--F led灯硬件结构 开关硬件结构…

SpringBoot整合Sharding-Jdbc实现分库分表和分布式全局id

SpringBoot整合Sharding-Jdbc Sharding-Jdbc sharding-jdbc是客户端代理的数据库中间件&#xff1b;它和MyCat最大的不同是sharding-jdbc支持库内分表。 整合 数据库环境 在两台不同的主机上分别都创建了sharding_order数据库&#xff0c;库中都有t_order_1和t_order_2两张…

移动安全威胁:今天和明天的危险

随着技术的发展&#xff0c;个人和公司的设备、数据和隐私所面临的威胁也在发生变化。在本文中&#xff0c;我们将仔细研究当今移动设备安全面临的主要威胁&#xff0c;并共同探讨不久的将来的前景。 但首先让我们从基础开始&#xff1a;如何对移动设备发起攻击&#xff1f; …

血的教训--redis被入侵之漏洞利用复现--总览

血的教训–redis被入侵之漏洞利用复现–总览 相信大家对于自己的服务器被入侵&#xff0c;还是比较憎恨的&#xff0c;我的就被攻击了一次&#xff0c;总结经验&#xff0c;自己也是整理了这一个系列&#xff0c;从最基础到最后面的自己总结被攻破的步骤&#xff0c;非常清晰的…

Jboss启动报错Unrecognized VM option PermSize=128m

1.问题现象 JBoss启动提示创建JAVA虚拟机失败&#xff0c;异常信息如下 异常截图 异常日志 .JBoss Bootstrap Environment .JBOSS_HOME: E:\Jboss\jboss-4.0.2_BR_5.4.4.21\bin\\.. .JAVA: C:\Java\jdk1.6.0_38\bin\java .JAVA_OPTS: -Xms64m -Xmx1024m -Dprogram.namerun.ba…