#!/usr/local/bin/python
#-*- coding: cp949 -*-
# date : 2006. 10. 15
#
# 스도쿠를 풀어주는 스크립트
import random
class Sodoku:
def __init__(self):
self.timer = 0
self.board = []
self.tmpBoard=[]
for y in range(0, 82):
self.board.append(0)
self.tmpBoard.append(511)
def display(self):
for y in range(0, 9):
for x in range(0, 9):
idx = y* 9 + x + 1;
if self.board[idx] != 0:
print "%d" %(self.board[idx]),
else:
print "_",
print '|',
print '\n'
def checkSquare(self, sx, sy):
sum = 0;
for y in range(0, 3):
for x in range(0, 3):
if ( int(self.board[ (y+sy)*9+(x+sx)+1]) > 0):
sum |= (1 << (int(self.board[ (y+sy)*9+(x+sx)+1]-1)))
return sum
def checkBoard(self):
ret = 0;
for y in range(0, 9):
sum = 0;
for x in range(0, 9):
if ( int(self.board[y*9+x+1]) > 0):
sum |= (1 << (int(self.board[y*9+x+1]-1)))
if (sum != 511):
ret = 1
for x in range(0, 9):
sum = 0;
for y in range(0, 9):
if ( int(self.board[y*9+x+1]) > 0):
sum |= (1 << (int(self.board[y*9+x+1]-1)))
if (sum != 511):
ret = 2
for y in range(0, 3):
for x in range(0, 3):
sum = self.checkSquare(x*3, y*3)
if (sum != 511):
ret = 3
return ret
def readData(self):
fp = open("sudoku.dat", 'r')
self.board = map(int, fp.read().split(','))
fp.close()
for y in range(0, 82):
self.tmpBoard[y] =511
def guessCount(self, num):
count = 0
for i in range(0, 9):
if ( (1 << i) & num ) != 0:
count += 1
return count
def findStartPos(self):
sx = 0
sy = 0
mvalue = 511
mcount = 10
for y in range(0,9):
for x in range(0,9):
if (self.board[y * 9 + x + 1] == 0):
temp = self.guessCount(self.tmpBoard[y*9 + x + 1] )
if (mcount > temp):
mvalue = self.tmpBoard[y*9 + x + 1]
mcount = temp
sx = x
sy = y
return (sx, sy, mvalue)
def isFull(self):
count = 0
for y in range(0,9):
for x in range(0,9):
if (self.board[y * 9 + x + 1] != 0):
count += 1
return count == 81
def solve(self):
if (self.checkBoard() != 0):
if (self.isFull()):
return 0
#get position
(x, y, num) = self.findStartPos()
# set number
for i in range(0, 9):
if ( (1 << i) & num ) != 0:
self.board[y*9 + x + 1] = i+1
# checkTempBoard
self.checkWidthTempBoard(x, y)
self.checkHeightTempBoard(x, y)
self.checkSquareTempBoard(x/3, y/3, x, y)
if (self.solve() == 1):
return 1
else:
self.unCheckWidthTempBoard(x, y)
self.unCheckHeightTempBoard(x, y)
self.unCheckSquareTempBoard(x/3, y/3, x, y)
self.board[y*9 + x + 1] = 0
return 0
else:
return 1
def checkTempBoard(self):
for y in range(0, 9):
for x in range(0, 9):
if (self.board[y * 9 + x + 1] != 0):
self.checkWidthTempBoard(x, y)
self.checkHeightTempBoard(x, y)
self.checkSquareTempBoard(x/3, y/3, x, y)
def checkWidthTempBoard(self, sx, sy):
for x in range(0,9):
if (self.board[sy * 9 + x + 1] == 0):
self.tmpBoard[sy * 9 + x + 1] &= ~(1 << (self.board[sy * 9 + sx + 1] - 1))
def checkHeightTempBoard(self, sx, sy):
for y in range(0,9):
if (self.board[y * 9 + sx + 1] == 0):
self.tmpBoard[y * 9 + sx + 1] &= ~(1 << (self.board[sy * 9 + sx + 1] - 1))
def checkSquareTempBoard(self, sx, sy, tx, ty):
for y in range(0,3):
for x in range(0,3):
if (self.board[(y + sy) * 9 + sx + x + 1] == 0):
self.tmpBoard[(y + sy * 3) * 9 + sx * 3 + x + 1] &= ~(1 << (self.board[ty * 9 + tx + 1] - 1))
def unCheckWidthTempBoard(self, sx, sy):
for x in range(0,9):
if (self.board[sy * 9 + x + 1] == 0):
self.tmpBoard[sy * 9 + x + 1] |= (1 << (self.board[sy * 9 + sx + 1] - 1))
def unCheckHeightTempBoard(self, sx, sy):
for y in range(0,9):
if (self.board[y * 9 + sx + 1] == 0):
self.tmpBoard[y * 9 + sx + 1] |= (1 << (self.board[sy * 9 + sx + 1] - 1))
def unCheckSquareTempBoard(self, sx, sy, tx, ty):
for y in range(0,3):
for x in range(0,3):
if (self.board[(y + sy) * 9 + sx + x + 1] == 0):
self.tmpBoard[(y + sy * 3) * 9 + sx * 3 + x + 1] |= (1 << (self.board[ty * 9 + tx + 1] - 1))
def generate(self):
self.readData()
self.checkTempBoard()
def main():
sodoku = Sodoku()
sodoku.generate()
sodoku.display()
sodoku.solve()
print "-"*60
sodoku.display()
#----------------------------------------------------------
# main
if __name__ == "__main__":
main()
0, 6, 0, 9, 0, 0, 0, 0, 4, 3, 0, 0, 0, 2, 9, 0, 7, 1, 0, 0, 0, 4, 0, 1, 0, 0, 5, 8, 5, 0, 0, 0, 0, 8, 0, 6, 0, 8, 0, 7, 1, 0, 9, 4, 0, 5, 0, 3, 0, 7, 0, 0, 0, 0, 2, 1, 5, 0, 0, 8, 0, 3, 0, 0, 0, 4, 8, 0, 6, 2, 0, 0, 0, 7, 2, 0, 0, 0, 0, 6, 0, 9
[실행결과]
6 | _ | 9 | _ | _ | _ | _ | 4 | 3 |
_ | _ | _ | 2 | 9 | _ | 7 | 1 | _ |
_ | _ | 4 | _ | 1 | _ | _ | 5 | 8 |
5 | _ | _ | _ | _ | 8 | _ | 6 | _ |
8 | _ | 7 | 1 | _ | 9 | 4 | _ | 5 |
_ | 3 | _ | 7 | _ | _ | _ | _ | 2 |
1 | 5 | _ | _ | 8 | _ | 3 | _ | _ |
_ | 4 | 8 | _ | 6 | 2 | _ | _ | _ |
7 | 2 | _ | _ | _ | _ | 6 | _ | 9 |
-----------------------------
6 | 1 | 9 | 8 | 7 | 5 | 2 | 4 | 3 |
3 | 8 | 5 | 2 | 9 | 4 | 7 | 1 | 6 |
2 | 7 | 4 | 6 | 1 | 3 | 9 | 5 | 8 |
5 | 9 | 2 | 4 | 3 | 8 | 1 | 6 | 7 |
8 | 6 | 7 | 1 | 2 | 9 | 4 | 3 | 5 |
4 | 3 | 1 | 7 | 5 | 6 | 8 | 9 | 2 |
1 | 5 | 6 | 9 | 8 | 7 | 3 | 2 | 4 |
9 | 4 | 8 | 3 | 6 | 2 | 5 | 7 | 1 |
7 | 2 | 3 | 5 | 4 | 1 | 6 | 8 | 9 |
'Coding > Python 삽질기' 카테고리의 다른 글
WxPython - HelloWorld (0) | 2008.12.22 |
---|---|
[조각코드] 숫자 배열 읽어서 리스트에 저장하는 코드 (0) | 2008.10.06 |
간단한 메모장2 (0) | 2008.09.17 |