python3.6数独问题的解决

作者:Jayshon_Jaa 时间:2022-06-21 20:40:32 

算法比较暴力,直接用穷举的方式一个一个去试,所以程序运行时间会比较长,运行时间视数独而定。
不过从一开始到运行成功,整个过程却是一波三折,设计算法就花了不少时间,然后就是不断地去调试,找bug。刚开始的时候为了省事直接在sudoku类中递归调用blank,但是老哥还是too young too simple,sometimes navie,计算量实在是太大了,后面编译器直接抛出 “RecursionError: maximum recursion depth exceeded while calling a Python object” 超过最大递归深度的错误。在把递归深度改到100000之后,又出现了堆栈溢出问题。当然,解决办法也是相当地暴力:把递归放入while循环中,一旦符合条件就直接exit(0),整个程序直接gg,然后退出结束。
当然,算法还可以再优化一下,可以不用那么暴力,先列出可能的值然后再填入,这样可以大大缩小整个程序的运行时间,但是……懒得优化了,就这样吧,又不是不能用(笑~)。

运行结果:

python3.6数独问题的解决

再试一个其他的数独:

python3.6数独问题的解决

这回就快得多了,11秒就完成了,比第一个数独不知高到哪里去了

代码如下所示:


import copy
import time

t1=time.time()
origin = [[8, 0, 0, 0, 0, 0, 0, 0, 0],
 [0, 0, 3, 6, 0, 0, 0, 0, 0],
 [0, 7, 0, 0, 9, 0, 2, 0, 0],
 [0, 5, 0, 0, 0, 7, 0, 0, 0],
 [0, 0, 0, 0, 4, 5, 7, 0, 0],
 [0, 0, 0, 1, 0, 0, 0, 3, 0],
 [0, 0, 1, 0, 0, 0, 0, 6, 8],
 [0, 0, 8, 5, 0, 0, 0, 1, 0],
 [0, 9, 0, 0, 0, 0, 4, 0, 0]]

class sudoku:
def debug(self): # 调试
for list in origin:
 print(list)
print("\n")

def check_repetition(self,list):#判断表中是否有重复值,0除外
flag=0
for i in range(1,10):
 if list.count(i)>=2:
 return 1
 else:
 flag=flag+1
if flag==9:
 return 0

def check_row(self,row):#检测横向是否有重复值,无则为返回0,有则返回1
list = origin[row] # 横向
r1 = self.check_repetition(list)
if r1 == 0:
 return 0
else :
 return 1

def check_column(self,column):#检测纵向是否重复值,无则为返回0,有则返回1
list = [] # 纵向
for num in origin:
 list.append(num[column])
r2 = self.check_repetition(list)
if r2==0:
 return 0
else:
 return 1

def check_square(self,x,y):#检测九宫格是否有重复值,无则为返回0,有则返回1
x,y=y,x
if x>=9 or y>=9:
 return
square = []#九宫格
for i in range(0+y//3*3, 3+y//3*3):
 for j in range(0+x//3*3, 3+x//3*3):
 square.append(origin[i][j])
r3 = self.check_repetition(square)
if r3==0:
 return 0
else:
 return 1

def check(self,x,y):#检测是否有重复值,无则为0,有则不为0
r1 = self.check_row(x)
r2 = self.check_column(y)
r3 = self.check_square(x, y)
result=r1+r2+r3
return result

def get_next(self): # 获得下一个空值,返回row,column值
i = 0
for list in origin:
 try: # 当0不在列表中时,跳过
 column = list.index(0)
 row = origin.index(list)
 res = (row, column)
 return res
 except ValueError:
 i = i + 1
 if i == 9:
  t2=time.time()
  print("总用时={}".format(t2 - t1))
  exit(0)

def poi(self,row, column): # 位置修正
if row == 0 and column == -1:
 return
if row == 8 and column == 9:
 return
if column == -1:
 column = 8
 row = row - 1
if column == 9:
 column = 0
 row = row - 1
return (row, column)

def get_last(self,row, column):
origin[row].insert(column, 0)
origin[row].pop(column + 1)
column = column - 1 # 获得上一个已填值的行、列位置
row, column = self.poi(row, column)#位置修正
r = origin[row][column] * compare[row][column]
while r != 0:
 column = column - 1
 row, column = self.poi(row, column)
 r = origin[row][column] * compare[row][column]
return (row, column)

def blank(self):
try:
 row,column=self.get_next()
except TypeError:#已填完
 exit(0)
j=0
flag=0
for i in range(1,10):
 origin[row].insert(column,i)
 origin[row].pop(column+1)
 self.debug()
 r = self.check(row, column)
 if r==0:#无重复值
 return
 else:
 j = j + 1
 if j==9:
  flag=1
  break
if flag==1:
 row, column = self.get_last(row, column)
 value=origin[row][column]
 self.debug()
 while value == 9:
 row, column = self.get_last(row, column)
 value = origin[row][column]
 self.debug()
 while value<9:
 for k in range(value+1,10):
  origin[row].insert(column, k)
  origin[row].pop(column + 1)
  self.debug()
  r=self.check(row,column)
  if r!=0:#有重复
  if k==9:
   row, column = self.get_last(row, column)
   value=origin[row][column]
   self.debug()
   while value==9:
   row, column = self.get_last(row, column)
   value = origin[row][column]
   self.debug()
   break
  else:
  return

if __name__=="__main__":
compare = copy.deepcopy(origin)
sudoku = sudoku()
while 1:
sudoku.blank()

来源:https://blog.csdn.net/Jayshon_Jaa/article/details/83040821

标签:python,python3.6,数独
0
投稿

猜你喜欢

  • SQL 字母数字混合型字段 按里面的数字排序

    2010-04-23 18:18:00
  • python 寻找list中最大元素对应的索引方法

    2021-02-16 07:37:52
  • 使用MySQL内建复制功能

    2009-12-15 21:36:00
  • TensorFlow模型保存和提取的方法

    2022-04-30 05:19:54
  • Matplotlib使用Cursor实现UI定位的示例代码

    2022-04-18 15:27:13
  • python实现简单日志记录库glog的使用

    2023-01-07 23:14:40
  • Python中Turtle库改变画笔(海龟)方向的两种方法总结

    2022-04-21 11:09:52
  • Python的动态重新封装的教程

    2023-08-23 15:26:39
  • Python动态可视化模块Pynimate初体验

    2021-03-22 16:35:09
  • python基础教程之简单入门说明(变量和控制语言使用方法)

    2023-01-06 00:25:20
  • asp中获取当前页面的地址与参数的函数代码

    2011-02-20 10:37:00
  • Python安装第三方库的3种方法

    2022-02-03 03:10:47
  • python取数作为临时极大值(极小值)的方法

    2021-02-27 10:32:01
  • Python 基于xml.etree.ElementTree实现XML对比示例详解

    2022-02-24 12:25:53
  • 用pandas中的DataFrame时选取行或列的方法

    2023-05-15 07:43:04
  • Mootools 1.2教程(13)——正则表达式

    2008-12-07 20:25:00
  • 在VSCode中配置PHP开发环境的实战步骤

    2023-06-06 02:13:06
  • 给验证码增加干扰的杂点

    2008-05-16 11:34:00
  • 实例讲解Python中函数的调用与定义

    2022-10-25 02:47:55
  • python显示天气预报

    2022-04-22 23:46:38
  • asp之家 网络编程 m.aspxhome.com