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

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

数字 1-9 在每一行只能出现一次。
数字 1-9 在每一列只能出现一次。
数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。

空白格用 ‘.’ 表示。
Note:

给定的数独序列只包含数字 1-9 和字符 '.' 。
你可以假设给定的数独只有唯一解。
给定数独永远是 9x9 形式的。

第一次的代码:
首先设置一个函数,判断数字i是否可以插入,然后设置递归函数,在’ . '处插入数字,当9x9表格被填满后,返回表格x

A=['1','2','3','4','5','6','7','8','9']
        def buchongtu(x,s,sx,sy):
            for i in range(9):
                if x[sy][i]==s: #判断行
                    return False
                if x[i][sx]==s: #判断列
                    return False
            a=3*(sx//3+1)
            b=3*(sy//3+1)
            for i in range(a-3,a): #判断所属3x3表格
                for j in range(b-3,b):
                    if x[j][i]==s:
                        return False
            return True #可行返回True
        
        def jsd(X,s,sx,sy):
            x=X[:] #避免直接修改表格,选择使用切片
            x[sy][sx]=s #将s填入表格
            while x[sy][sx] in A: #当此格是数字时,不可填
                sx+=1             #遍历到不是数组的格子
                if sx==9: #遍历到行尾
                    sy=sy+1
                    sx=0
                if sy==9: #遍历完表格,返回答案
                    return x
            for i in A: #从1-9填写此空格
                if buchongtu(x,i,sx,sy): #如果数字可行
                    jsd(x,i,sx,sy) #迭代
        
        return jsd(board,board[0][0],0,0)

但是没解出来,有可能是在递归过程中,每一个数独解都是从1-9遍历,有一个解猜错误之后,会不断返回一个NONE,导致程序提前结束
第二次代码:
修改了jsd函数,却忘记修好buchongtu函数,导数我一直没有得到答案,因为在jsd函数中已经对x[sy][sx]赋值,如果不修改,buchongtu函数会一直返回False。在jsd函数中,在添加数字的同时判断是否是正确数字,因为只有一个解,在递归过程中,已经把数字全部填写好了。

A=['1','2','3','4','5','6','7','8','9']
        def buchongtu(x,s,sx,sy):
            for i in range(9):
                if x[sy][i]==s and i!=sx: #防止选择sx列
                    return False
                if x[i][sx]==s and i!=sy: #防止选择sy行
                    return False
            a=3*(sx//3+1)
            b=3*(sy//3+1)
            for i in range(a-3,a): #判断3x3表格
                for j in range(b-3,b):
                    if x[j][i]==s and j!=sy and i!=sx:
                        return False  #防止选择[sy][sx]坑
            return True
        
        def jsd(x):
            for i in range(9):
                for j in range(9):
                    if x[i][j]=='.': #对空操作
                        for k in A:
                            x[i][j]=k #填写k
                            if buchongtu(x,k,j,i) and jsd(x):
                                return True #如果正确,返回True
                            x[i][j]='.' #如果k不正确,则先不填写,等重填
                        return False #如果没有返回True,表明没有数字可填
                                     #第一个数字就填错了,则返回False,阻止
                                     #第一个数字的填写,只有全部正确才能填
            return True #当两个for不返回值时,表明已经遍历结束,刚才填写正确       
        
        jsd(board)

908ms,排名50%
如果jsd()不采用两个for循环,可采用while:
作用是一样的遍历整个列表的空

def jsd(x,index):
            while index <81:#有点巧妙,计算行i,列j
                i=index/9
                j=index%9
                if x[i][j]=='.':
                    for k in A:
                        x[i][j]=k
                        if buchongtu(x,k,j,i) and jsd(x,index):
                            return True
                        x[i][j]='.'
                    return False
                index+=1
            return True        
        
        jsd(board,0)

676ms,排名57%
当然也可以while结束后再for循环:

def jsd(x,index):
            while index <81:
                i=index/9
                j=index%9
                if x[i][j]=='.':
                    break #反正都是每次递归都会填写一个数
                index+=1  #所以每次调用只需要找一个空就行了  
            
            if index<81:#如果没满
                for k in A:
                    x[i][j]=k
                    if buchongtu(x,k,j,i) and jsd(x,index+1):#这里+1可去
                        return True #确保填写的是正确的
                    x[i][j]='.'
                return False #如无法填写,返回重填
            else: #如果满了
                return True        
        
        jsd(board,0)

608ms,排名58%