python解数独谜题思路和代码分享

2023-10-16 22:50

本文主要是介绍python解数独谜题思路和代码分享,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

数独简介

         数独(shù dú)是源自18世纪瑞士的一种数学游戏。玩家需要根据9×9盘面上的已知数字,推理出所有剩余空格的数字,并满足每一行、每一列、每一个粗线宫(3*3)内的数字均含1-9,不重复。

数独基础规则1

        数独的每个单元格内容要满足每一行、每一列、每一个粗线宫(3*3)内的数字均含1-9,不重复。根据这条规则,我们能很容易得出以下判断方法:

 # 基础校验规则,检验num是否符合行列九宫格三项def base_rule(row, col, num):# 检查行for i in range(n):if board[row][i] == num:return False# 检查列for i in range(n):if board[i][col] == num:return False# 检查 3x3 方格内是否存在相同数字start_row = 3 * (row // 3)start_col = 3 * (col // 3)for i in range(3):for j in range(3):if board[start_row + i][start_col + j] == num:return Falsereturn True

 如果我们填入了一个值,他满足行,列,九宫格都未出现过,那么他便是这个单元格中的一个“可能值“

 可能值规则

        什么是可能值?对于一个数独的空位置来说,没有到达一定的深度之前,我们这个空位使用规则一能满足的值便是可能值,一个空位置里,有一个或多个可能值的存在

[[5], [3], [1, 2, 4], [2, 4], [7], [4, 8], [1, 4, 9], [2, 8], [1, 2, 4]]
[[6], [2, 4, 7], [2, 4, 7], [1], [9], [5], [3, 4, 7], [2, 3, 7, 8], [2, 3, 4]]
[[1, 2, 4], [9], [8], [2, 4], [2], [3, 4], [1, 3, 4, 5, 7], [6], [1, 2, 3, 4, 5]]
[[8], [2], [1, 2, 5], [1, 4, 5, 7, 9], [6], [1, 4, 5, 9], [4, 5, 7, 9], [2, 7], [3]]
[[4], [2], [2, 5], [8], [5, 7, 9], [3], [5, 7, 9], [2, 7], [1]]
[[7], [3], [1, 3, 5], [1, 4, 5, 9], [2], [1, 4, 5, 9], [4, 5, 9], [8], [6]]
[[1, 4, 9], [6], [1, 3, 4, 5, 7], [5, 7], [5, 7], [3, 5], [2], [8], [1, 3, 4]]
[[2, 8], [2, 3, 7, 8], [2, 3, 7], [4], [1], [9], [3], [3, 6], [5]]
[[1, 2, 4], [2, 3, 4], [1, 2, 3, 4, 5], [2, 5], [8], [3, 5], [1, 3, 4], [7], [9]]

图1、一个数独内可能值举例

这样看可能会比较抽象,但是我们结合原始数独来分析:

5 3 0 | 0 7 0 | 0 0 0   

6 0 0 | 1 9 5 | 0 0 0

0 9 8 | 0 0 0 | 0 6 0

---------------------

8 0 0 | 0 6 0 | 0 0 3

4 0 0 | 8 0 3 | 0 0 1

7 0 0 | 0 2 0 | 0 0 6

---------------------

0 6 0 | 0 0 0 | 2 8 0

0 0 0 | 4 1 9 | 0 0 5

0 0 0 | 0 8 0 | 0 7 9

比如我们看第一行的第三个,[1,2,4]是该空位的三个可能值,因为其他数字在行列九宫格出现过,所以这一空我们只能在这三个中选择其中之一,那么该如何得出可能值呢?

 # 消除规则一,经过base_rule筛选后的可能值def rule_1():list_3d = [[[] for _ in range(9)] for _ in range(9)]for row in range(9):for col in range(9):if board[row][col] != 0:list_3d[row][col].append(board[row][col])else:for i in range(1, 10):if base_rule(row, col, i):list_3d[row][col].append(i)# print("规则一之后")# for row in range(9):#     print(list_3d[row])#     print()return list_3d

board是我们的原始数组,我创建的list_3d类是一个二维列表,这个二维列表的每个元素又是一个列表,从图1我们便可以看出。使用if判断不为零的直接放入,而为0的(空位置)则需要用base_rule循环判断,打印列表,你也能得到类似图一的效果。

消除规则二

        原理:如果可能某个值在行、列、九宫格里只出现过一次,那么我们便可以判定这个空位置的值就是这个只出现过一次的值。

这个原理也可以反向理解,如果一个可能值只在这个位置出现了,而行列九宫格中的其他地方没有,是不是表示这个值就只能待在这个空,否则他无处可去了呢?

通过这项规则,我们似乎又可以确定一些空位置该填什么了:

 # 基础消除规则2,如果可能值满足在行列九宫只出现1次,则可以确定为该值def base_rule2(row, col, num, current_board):flag1 = Trueflag2 = Trueflag3 = Truefor i in range(n):if i != col and num in current_board[row][i]:flag1 = Falsebreak# 检查列for i in range(n):if i != row and num in current_board[i][col]:flag2 = Falsebreak# 检查 3x3 方格内是否存在相同数字start_row = 3 * (row // 3)start_col = 3 * (col // 3)for i in range(start_row, start_row + 3):if not flag3:breakfor j in range(start_col, start_col + 3):if num in current_board[i][j] and (i != row or j != col):flag3 = Falsebreak# if flag1 or flag2 or flag3:# print("检查到唯一值" + str(row) + "," + str(col) + "," + str(num) + str(flag1) + ",# " + str(flag2) + "," + str(flag3))return flag1 or flag2 or flag3

并且,我们还能通过确认这个值,再次消除一些可能值,假如[1,2,4]这三个里,我确认了1是唯一的,那么除了他所在的那个单元格,与他同行,同列,同九宫格的所有的1都可以被消除:

 # 通过规则二检测到可能值后,清除行,列,九宫格内所有可能值,被间接消除的只剩一个是会继续调用def clear_row(row, col, num, current_board):for i in range(9):if i != col and num in current_board[row][i]:current_board[row][i].remove(num)if len(current_board[row][i]) == 1:clear_col(row, i, current_board[row][i][0], current_board)clear_cell(row, i, current_board[row][i][0], current_board)# print("得到值,继续消除列和九宫" + str(current_board[row][i][0]))def clear_col(row, col, num, current_board):for i in range(9):if i != row and num in current_board[i][col]:current_board[i][col].remove(num)if len(current_board[i][col]) == 1:clear_row(i, col, current_board[i][col][0], current_board)clear_cell(i, col, current_board[i][col][0], current_board)# print("得到值,继续消除行和九宫" + str(current_board[i][col][0]))def clear_cell(row, col, num, current_board):start_row = 3 * (row // 3)start_col = 3 * (col // 3)for i in range(start_row, start_row + 3):for j in range(start_col, start_col + 3):if num in current_board[i][j] and (i != row or j != col):current_board[i][j].remove(num)if len(current_board[i][j]) == 1:clear_row(i, j, current_board[i][j][0], current_board)clear_col(i, j, current_board[i][j][0], current_board)# print("得到值,继续消除行和列" + str(current_board[i][j][0]))

注意,我为什么说间接消除的要多执行一次消除?比如 [1,5]......[2,5]在同一行,当遍历到[2,5]时,确认可能值只有5,那么[1,5]中会被消除的只剩[1],相当于又确认了一个值,所以我们要移到[1,5]所在的行列继续进行消除

搜索+回溯

        经过一次消除规则1和消除规则2的处理,我们数独空位里的可能值也少了很多,这个时候我们可以选择直接遍历+回溯带入可能值来进行最终的解数独,也可以使用搜索+回溯来解,这里我选择的是搜索+回溯,每次搜索可能值最少的那个位置开始试

  def search(result):current_len = 9999current_row = 9999current_col = 9999for row in range(9):for col in range(9):if current_len >= len(result[row][col]) > 1:current_len = len(result[row][col])current_row = rowcurrent_col = colreturn current_row, current_col# 经过一二消除后,开始递归可能值解数独def recursion(result):if isOver(result):return Truerow, col = search(result)for i in result[row][col]:if board[row][col] == 0 and base_rule(row, col, i):temp = copy.deepcopy(result[row][col])result[row][col] = [i]board[row][col] = iif recursion(result):return True# 撤销选择,将之前移除的候选数字添加回去并重置为0result[row][col] = tempboard[row][col] = 0return False

到这里解数独大部分程序都完成了,最后根据自己需要,选择输入值,判断数独结束方法等

最终执行效果:

 解50个大约14s,简单数独平均0.01s一个,但是这50个里有两个解的很慢,一个2s,一个8s,这点笔者也还没研究出是为什么

 

经过分析后发现是搜索方法的问题,注释掉搜索方法,重新编写回溯执行,只需要2s

这篇关于python解数独谜题思路和代码分享的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

一文教你Python如何快速精准抓取网页数据

《一文教你Python如何快速精准抓取网页数据》这篇文章主要为大家详细介绍了如何利用Python实现快速精准抓取网页数据,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录1. 准备工作2. 基础爬虫实现3. 高级功能扩展3.1 抓取文章详情3.2 保存数据到文件4. 完整示例

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

基于Python打造一个智能单词管理神器

《基于Python打造一个智能单词管理神器》这篇文章主要为大家详细介绍了如何使用Python打造一个智能单词管理神器,从查询到导出的一站式解决,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 项目概述:为什么需要这个工具2. 环境搭建与快速入门2.1 环境要求2.2 首次运行配置3. 核心功能使用指

Python实现微信自动锁定工具

《Python实现微信自动锁定工具》在数字化办公时代,微信已成为职场沟通的重要工具,但临时离开时忘记锁屏可能导致敏感信息泄露,下面我们就来看看如何使用Python打造一个微信自动锁定工具吧... 目录引言:当微信隐私遇到自动化守护效果展示核心功能全景图技术亮点深度解析1. 无操作检测引擎2. 微信路径智能获

Python中pywin32 常用窗口操作的实现

《Python中pywin32常用窗口操作的实现》本文主要介绍了Python中pywin32常用窗口操作的实现,pywin32主要的作用是供Python开发者快速调用WindowsAPI的一个... 目录获取窗口句柄获取最前端窗口句柄获取指定坐标处的窗口根据窗口的完整标题匹配获取句柄根据窗口的类别匹配获取句

利用Python打造一个Excel记账模板

《利用Python打造一个Excel记账模板》这篇文章主要为大家详细介绍了如何使用Python打造一个超实用的Excel记账模板,可以帮助大家高效管理财务,迈向财富自由之路,感兴趣的小伙伴快跟随小编一... 目录设置预算百分比超支标红预警记账模板功能介绍基础记账预算管理可视化分析摸鱼时间理财法碎片时间利用财

Python中的Walrus运算符分析示例详解

《Python中的Walrus运算符分析示例详解》Python中的Walrus运算符(:=)是Python3.8引入的一个新特性,允许在表达式中同时赋值和返回值,它的核心作用是减少重复计算,提升代码简... 目录1. 在循环中避免重复计算2. 在条件判断中同时赋值变量3. 在列表推导式或字典推导式中简化逻辑

python处理带有时区的日期和时间数据

《python处理带有时区的日期和时间数据》这篇文章主要为大家详细介绍了如何在Python中使用pytz库处理时区信息,包括获取当前UTC时间,转换为特定时区等,有需要的小伙伴可以参考一下... 目录时区基本信息python datetime使用timezonepandas处理时区数据知识延展时区基本信息

Python位移操作和位运算的实现示例

《Python位移操作和位运算的实现示例》本文主要介绍了Python位移操作和位运算的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 位移操作1.1 左移操作 (<<)1.2 右移操作 (>>)注意事项:2. 位运算2.1

使用Python和Pyecharts创建交互式地图

《使用Python和Pyecharts创建交互式地图》在数据可视化领域,创建交互式地图是一种强大的方式,可以使受众能够以引人入胜且信息丰富的方式探索地理数据,下面我们看看如何使用Python和Pyec... 目录简介Pyecharts 简介创建上海地图代码说明运行结果总结简介在数据可视化领域,创建交互式地