1

我在使用 Python 编程方面相对较新。这段代码运行良好,直到我尝试将它变成一个类。我正在为我的数独求解器代码开设课程,练习课程并涉足面向对象的编程。

因此,我从有类似问题的用户那里阅读了一大堆问题,大多数答案是: - 在从所述类调用函数之前先实例化类,但它们似乎都不适用于我的具体示例。

这是我的课:

#assume sudoku is imported as a np.array, white spaces replaced by zeros


class Sudoku():

    solution_number = 1

    def __init__ (self, sud_arr):
        self.sudo = sud_arr

    #print(self.sudo)

    def possible (self, y, x, num):
        for i in range(9):
            if self.sudo[y][i] == num:
                return False
            if self.sudo[i][x] == num:
                return False
            yy = (y//3)*3
            xx = (x//3)*3
            for i in range(3):
                for j in range(3):
                    if self.sudo[yy+i][xx+j] == num:
                        return False
        return True


    def solve(self):
        for i in range(9):
            for j in range(9):
                if self.sudo[i][j] == 0:
                    for nr in range(1,10):
                         if Sudoku.possible(i,j,nr): #line 34
                            self.sudo[i][j] = nr
                            Sudoku.solve()
                            self.sudo[i][j] = 0
                    return
        if Sudoku.solution_number > 1:  #if there is more than one solution, include solution number
            print("Solution Number {}".format(Sudoku.solution_number))
        else: print("Solution Number 1")
        print(self.sudo)                                  
        Sudoku.add_sol_num()

    @classmethod
    def add_sol_num(cls):           
        cls.solution_number += 1

运行后:

s = Sudoku(su) #where su is a numpy array sudoku
s.solve() #line 52

我得到错误:

  File "/Users/georgesvanheerden/Python/Projects/Sudoku/SudokuSolver.py", line 52, in <module>
    s.solve()
  File "/Users/georgesvanheerden/Python/Projects/Sudoku/SudokuSolver.py", line 34, in solve
    if Sudoku.possible(i,j,nr):
TypeError: possible() missing 1 required positional argument: 'num'
[Finished in 1.9s with exit code 1]

抱歉,如果代码太多,我不知道要删掉哪些部分。

4

4 回答 4

1

使用self.possible方法时,Sudoku.possible 会为您提供对该方法的引用,该方法无法找到您从中调用它的实例。

这也适用于if Sudoku.solution_number > 1,通常pythonic的方式是使用self变量,或者方法的第一个参数(虽然你也可以传递self给函数Solution.possible(self, i, j , nr):)

所以你的代码看起来像:

    def solve(self):
        for i in range(9):
            for j in range(9):
                if self.sudo[i][j] == 0:
                    for nr in range(1,10):
                         if self.possible(i,j,nr): #line 34
                            self.sudo[i][j] = nr
                            self.solve()
                            self.sudo[i][j] = 0
                    return
        if self.solution_number > 1:  #if there is more than one solution, include solution number
            print("Solution Number {}".format(self.solution_number))
        else: print("Solution Number 1")
        print(self.sudo)                                  
        Sudoku.add_sol_num() # add_sol_num is a @classmethod
于 2020-06-19T10:31:24.847 回答
0

您可以添加 self 作为第一个参数:

if Sudoku.possible(self, i, j, nr):  #line 34
于 2020-06-19T10:34:00.117 回答
0

让我们首先了解给出的错误:

TypeError: possible() missing 1 required positional argument: 'num'

num这表示调用possible()方法时参数没有价值。

但是您在这里传递了 3 个参数:

if Sudoku.possible(i,j,nr)

那么这里出了什么问题???

如果您看到方法的定义:

def possible (self, y, x, num):

这表示您将传递 4 个参数,其中一个是类的实例/对象(self参数)。

1. If we invoke this method using a class object, `self` argument is passed by default. So in this case we can just send 3 arguments apart from `self`.

2. If you want to invoke this method like you have done above, you will have to provide a value for  `self` argument explicitally.

所以,你可以这样做(pythonic 方式和好方法):在调用可能的方法时,使用self关键字。

if self.possible(i,j,nr):
于 2020-06-19T10:45:39.467 回答
0

在第 34 行和第 36 行中,您调用了两个方法,就好像它们是静态方法一样,因为您调用的是类上的方法而不是某个实例上的方法。这也是为什么self不被识别并因此在方法调用中要求另一个参数的原因。你想调用当前数独实例的方法。所以

if self.possible(i,j,nr):

在第 34 行和

self.solve()

在第 36 行应该可以解决问题。

于 2020-06-19T11:40:57.577 回答