queens.py 2.22 KB
Newer Older
1
#!/usr/bin/env python3
2

3 4
"""
N queens problem.
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21

The (well-known) problem is due to Niklaus Wirth.

This solution is inspired by Dijkstra (Structured Programming).  It is
a classic recursive backtracking approach.
"""

N = 8                                   # Default; command line overrides

class Queens:

    def __init__(self, n=N):
        self.n = n
        self.reset()

    def reset(self):
        n = self.n
22 23
        self.y = [None] * n             # Where is the queen in column x
        self.row = [0] * n              # Is row[y] safe?
24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
        self.up = [0] * (2*n-1)         # Is upward diagonal[x-y] safe?
        self.down = [0] * (2*n-1)       # Is downward diagonal[x+y] safe?
        self.nfound = 0                 # Instrumentation

    def solve(self, x=0):               # Recursive solver
        for y in range(self.n):
            if self.safe(x, y):
                self.place(x, y)
                if x+1 == self.n:
                    self.display()
                else:
                    self.solve(x+1)
                self.remove(x, y)

    def safe(self, x, y):
        return not self.row[y] and not self.up[x-y] and not self.down[x+y]

    def place(self, x, y):
        self.y[x] = y
        self.row[y] = 1
        self.up[x-y] = 1
        self.down[x+y] = 1

    def remove(self, x, y):
        self.y[x] = None
        self.row[y] = 0
        self.up[x-y] = 0
        self.down[x+y] = 0

53
    silent = 0                          # If true, count solutions only
54 55 56 57 58

    def display(self):
        self.nfound = self.nfound + 1
        if self.silent:
            return
59
        print('+-' + '--'*self.n + '+')
60
        for y in range(self.n-1, -1, -1):
61
            print('|', end=' ')
62 63
            for x in range(self.n):
                if self.y[x] == y:
64
                    print("Q", end=' ')
65
                else:
66 67 68
                    print(".", end=' ')
            print('|')
        print('+-' + '--'*self.n + '+')
69 70 71 72 73 74 75 76 77 78 79 80 81

def main():
    import sys
    silent = 0
    n = N
    if sys.argv[1:2] == ['-n']:
        silent = 1
        del sys.argv[1]
    if sys.argv[1:]:
        n = int(sys.argv[1])
    q = Queens(n)
    q.silent = silent
    q.solve()
82
    print("Found", q.nfound, "solutions.")
83 84 85

if __name__ == "__main__":
    main()