递归算法学习——有效的数独,解数独

news2025/1/9 21:18:51

一,有效的数独

1.题意

请你判断一个 9 x 9 的数独是否有效。只需要 根据以下规则 ,验证已经填入的数字是否有效即可。

  1. 数字 1-9 在每一行只能出现一次。
  2. 数字 1-9 在每一列只能出现一次。
  3. 数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。(请参考示例图)

注意:

  • 一个有效的数独(部分已被填充)不一定是可解的。
  • 只需要根据以上规则,验证已经填入的数字是否有效即可。
  • 空白格用 '.' 表示。

2.解释

有效的数独这道题的其实是不需要用到递归的。这道题其实就是一个判断题,要保证的的便是再一个9*9大小的二维数组里1~9这九个数字在每一行每一列每一个九宫格里面只出现了一次。如以下例子:

黑实线分割开的就是一个九宫格。这个数独的每一行每一列出现的数据都是唯一的,所以这个数独是有效的。

3.题目接口

class Solution {
public:
    bool isValidSudoku(vector<vector<char>>& board) {

    }
};

4.解题思路及代码

要解决这道题,我们首先就需要三个数组。这三个数组记录的便是我们是否在某一行,某一列,某一个九宫格里面是否又出现过某一个数字。在遍历过程中若出现了某一个数字便将这一行,这一列,这一个九宫格的第出现的数字的这一个位置标记为true。下次如果还会遍历同样的数字便返回false。如下代码:

class Solution {
public:
    bool row[9][10];
    bool col[9][10];
    bool grid[3][3][10];
    bool isValidSudoku(vector<vector<char>>& board) {
        for(int i = 0;i<9;i++)
        {
            for(int j = 0;j<9;j++)
            {
                if(board[i][j]!='.')//当遍历到的数据不是'.'时便可以进入判断
                {
                    int num = board[i][j]-'0';//将字符转换为数字
                    if(row[i][num]||col[j][num]||grid[i/3][j/3][num])//当这个数字在行,列,九宫格任何一个位置上出现时便可以返回false
                    {
                        return false;
                    }
                     row[i][num] = col[j][num] = grid[i/3][j/3][num] = true;//遍历过后将这一行这一列这个九宫格上的这个数字记录下来
                }
            }
        }

        return true;
    }
};

在这里大家可能比较疑惑的便是grid[i/3][j/3][num]了。其实这里便是将九宫格坐标化了,当行数和列数都在1~3时对标的便是下标0,4~6对标的便是下标1,6~8对标的便是下标2。在一个9*9的格子里面有九个九宫格,按照上面的分法下标分别是(0,0),(0,1),(0,2),(1,0),(1,1),(1,2),(2,0),(2,1),(2,2)。

二,解数独

1.题意

编写一个程序,通过填充空格来解决数独问题。

数独的解法需 遵循如下规则

  1. 数字 1-9 在每一行只能出现一次。
  2. 数字 1-9 在每一列只能出现一次。
  3. 数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。(请参考示例图)

数独部分空格内已填入了数字,空白格用 '.' 表示。

2.解释

这一道题便是让我们来填入数字来解决数独问题了。上面的有效的数独不需要用到递归的方式来解决但是这道题便要使用递归来解决了。

3.题目接口

class Solution {
public:
    void solveSudoku(vector<vector<char>>& board) {

    }
};

4.解题思路及代码

先将代码写出再来解释,代码:

class Solution {
public:
    bool row[9][10];
    bool col[9][10];
    bool grid[3][3][10];

    void solveSudoku(vector<vector<char>>& board) {
        for(int i = 0;i<9;i++)//先将数独里面出现的数字标记下来
        {
            for(int j = 0;j<9;j++)
            {
                if(board[i][j]!='.')
                {
                    int num = board[i][j]-'0';
                    row[i][num] = col[j][num] = grid[i/3][j/3][num] = true;
                }
            }
        }  
        dfs(board); 
    }

    bool dfs(vector<vector<char>>&board)
    {
        for(int i = 0;i<9;i++)
        {
            for(int j = 0;j<9;j++)
            {
                  if(board[i][j] == '.')
                  {
                      for(int num = 1;num<=9;num++)
                      {
                          if(!row[i][num]&&!col[j][num]&&!grid[i/3][j/3][num])
                          {
                              board[i][j] = num+'0';
                              row[i][num] = col[j][num] = grid[i/3][j/3][num] = true;
                              if(dfs(board)) return true;
                              board[i][j] = '.';//当走到这里时便是因为这一层填的数字的不到结果所以要将这个位置的值改回'.'标记改为false。
                              row[i][num] = col[j][num] = grid[i/3][j/3][num] = false;

                          }
                      }
                      return false;//当遍历到的这一个格子九个数字都不能填时便返回false调整上一层的值
                  }
            }
        }
        return true;
    }
};

在这一道题里面最让人难以理解的便是没有递归出口,因为递归必须要有出口才能返回到上一层。但是这道题的代码里面似乎没有是吧。其实不是的,这道题只是没有显示的写出递归出口。它是使用两个for循环来隐式的作为递归出口了,当一个棋盘被遍历完了以后或者不能得到结果时便会返回到上一层重新操作。

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

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

相关文章

【LeetCode】83. 删除排序链表中的重复元素

83. 删除排序链表中的重复元素&#xff08;简单&#xff09; 方法&#xff1a;一次遍历 思路 由于给定的链表是排好序的&#xff0c;因此重复的元素在链表中出现的位置是连续的&#xff0c;因此我们只需要对链表进行一次遍历&#xff0c;就可以删除重复的元素。 从指针 cur 指…

MATLAB旋转动图的绘制

MATLAB旋转动图的绘制 文章目录 MATLAB旋转动图的绘制1、动图效果2、matlab代码 利用matlab实现三维旋转动图的绘制。 1、动图效果 2、matlab代码 close all clear clcf(x,y,z)(x.^2 (9./4).*y.^2 z.^2 - 1).^3 - x.^2.*z.^3 - (9./80).*y.^2.*z.^3; [x,y,z]meshgrid(linspac…

STM32 SPI对存储芯片发送写是能命令后一直忙等待

我采用CUBE配置的SPI外设&#xff0c;对NSS引脚选择了硬件输出&#xff0c;这种方式对读取命令没有影响&#xff0c;但是对写命令有&#xff0c;当我发送写是能命令后&#xff0c;读取状态寄存器的值一直都是忙&#xff0c;我猜测这可能是硬件控制NSS引脚后&#xff0c;对于HAL…

Java迷宫问题

迷宫问题&#xff1a; 问题&#xff1a; /*迷宫问题 &#xff1a; 让小球从左上角开始 到达右上角 绕过障碍物求最短路径这里*号是障碍物的意思* * * * * * ** ** * ** * * ** ** ** ** * * * * * *思路 &#xff…

Linux安装JDK并配置环境变量

1.下载jdk 可以去官网直接下载JDK&#xff0c;下载时选择好需要的版本和操作系统位数 2.安装jdk &#xff08;1&#xff09;选择合适的位置新建安装目录 mkdir -p /home/local/java&#xff08;2&#xff09;进入目录 cd /home/local/java&#xff08;3&#xff09;上传j…

2023年9月CDGA/CDGP数据治理认证考试报名,当然弘博创新

据DAMA中国官方网站消息&#xff0c;2023年度第三期DAMA中国CDGA和CDGP认证考试定于2023年9月23日举行。 报名通道现已开启&#xff0c;相关事宜通知如下&#xff1a; 考试科目: 数据治理工程师(CertifiedDataGovernanceAssociate,CDGA) 数据治理专家(CertifiedDataGovernanc…

电子半导体行业电能质量监测与治理系统解决方案 安科瑞 许敏

摘要&#xff1a;在国家鼓励半导体材料国产化的政策导向下&#xff0c;本土半导体材料厂商不断提升半导体产品技术水平和研发能力&#xff0c;逐渐打破了国外半导体厂商的垄断格局&#xff0c;推进中国半导体材料国产化进程&#xff0c;促进中国半导体行业的发展。半导体产品的…

如何让insert程序速度快,可以试试联合SQL(insert 和 select 一起使用)?

查询添加可选择SQL执行&#xff0c;速度远超程序执行 insert 和 select案例 insert into 表1(列1,列2,列3,...) select 列1,列2,列3,...from表2(GROUP BY 列)116511 条数据 耗时45秒&#xff0c; 如果是程序查询然后再insert&#xff0c;则需要30分钟左右&#xff01;&#x…

docker安装mysql8之lower_case_table_names参数

docker安装mysql8之lower_case_table_names参数 前言发现问题找到问题解决方案不挂载外部文件挂载外部文件 前言 既然使用docker安装mysql&#xff0c;看重的就是一键拉取和一键启动&#xff0c;然后在安装mysql容器的过程中&#xff0c;出现了新问题&#xff0c;关于lower_ca…

Python Asyncio 之网络编程方法详解

https://so1n.me/2023/08/29/python_asyncio_lib_network/

CS420 课程笔记 P8 - 如何编辑汇编代码 (最终篇)

文章目录 IntroHardwareReturn to virtual memoryCompilerFinding assembly codeExample!Assembly languageRegisters Assembly ExampleIncDecAddSubMov!NegMulDiv Assembly problem恭喜 Congratulations! Intro 大多数人在走到这里之前就退出了&#xff0c;这里是 game hacki…

nginx--技术文档--基本概念--《十分钟快速扫盲》

nginx是什么&#xff1f; Nginx是一个高性能的HTTP和反向代理web服务器&#xff0c;同时也提供了IMAP/POP3/SMTP服务。它具有高并发性、稳定性和灵活性&#xff0c;并且使用事件驱动的方式处理请求&#xff0c;能够有效地处理大量并发连接。此外&#xff0c;Nginx还具有高效的内…

总结开发中一些数据处理方法的封装

摘要&#xff1a; 开发中经常会遇到一些组件需要的特定数据结构&#xff0c;后端不一定会返回你需要的数据结构的&#xff0c;所以还是要前端来处理的&#xff01;这里来总结一下平常开发中遇到的需要处理结构的方法&#xff0c;下次遇到直接拿来用就可以了&#xff01; 目录概…

云架构师学习------技术路线与总结

云架构师学习------技术路线与总结 云架构师学习------技术路线与总结一、什么是架构IT架构-数据架构-应用架构IT架构应用架构数据架构 架构的六个层面基础设施层数据层中间层基础服务层业务服务层用户接口层 二、云计算的历史演进与基本原理云计算的本质&#xff1a;资源到架构…

小度音响 小众安卓手机 个别车机等系列mtk芯片刷机 修改 导出系统 root等操作解析

目前很多机型采用的是mtk芯片。包括小度1C.个别车机 或者其他安卓设备。这类机型很少有官方固件或者未曾流出。有的机型限制刷机 root 安装软件等等。那么类似的设备有没有方法来root或者导出系统呢。其实前面几篇博文中我有介绍如何提取备份mtk芯片的系统。今天在这些博文的基…

企业文件数据透明加密保护——防泄密软件系统

天锐绿盾电脑文件数据透明加密、防泄密系统是一款全面的加密软件系统&#xff0c;主要从源头上保障数据安全和使用安全。该系统采用文件过滤驱动实现透明加解密&#xff0c;对用户完全透明&#xff0c;不影响用户操作习惯&#xff0c;从源头上保障企业数据安全。通过对电子文档…

ajax day2

1、 2、控制弹框显示和隐藏&#xff1a; 3、右键tr&#xff0c;编辑为html&#xff0c;可直接复制tr部分的代码 4、删除时&#xff0c;点击删除按钮&#xff0c;可以获取图书id&#xff1a; 5、编辑图书 快速赋值表单元素内容&#xff0c;用于回显&#xff1a; 6、hidden …

python开发之个微机器人的二次开发

简要描述&#xff1a; 取消消息接收 请求URL&#xff1a; http://域名地址/cancelHttpCallbackUrl 请求方式&#xff1a; POST 请求头Headers&#xff1a; Content-Type&#xff1a;application/json 参数&#xff1a; 参数名类型说明codestring1000成功&#xff0c;1…

MySql学习笔记12——数据库设计三范式

数据库设计三范式 第一范式&#xff1a;要求任何一张表必须有主键&#xff0c;每一个字段原子性不可再分。 第二范式&#xff1a;建立在第一范式之上&#xff0c;要求所有非主键字段必须完全依赖主键&#xff0c;不能部分依赖 第三范式&#xff1a;建立在第二范式之上&#…

分享2款微课录制软件,保证让你满意!

“录微课用什么软件呀&#xff0c;真的服了&#xff0c;平台自带的录屏画质太差了&#xff0c;完全看不清讲的内容&#xff0c;而且音质也不是很好&#xff0c;大家有没有微课录制的软件推荐&#xff0c;谢谢啦” 随着教育方式的转型和技术的发展&#xff0c;微课程成为了一种…