前言
使用python简单实现一下数独小游戏,并且使用tkinter进行界面展示。
一、题目生成
1.数独规则
- 在 9x9 的棋盘网格中将数字 1 ~ 9 填入空白格
- 每一列只能包含数字 1 到 9,不能重复
- 每一行只能包含数字 1 到 9,不能重复
- 每个 3×3 的小九宫格只能包含数字 1 到 9,每一列或每一行中的每个数字只能使用一次
2.生成初始题目
先直接得到若干个数独游戏的答案,然后再随机让一些数字变成待填入的空白格就OK了,而随机变成空白格的数量就决定着游戏难度的大小。
使用递归的方法实现,并且在create_board函数
返回答案和题目,来看代码:
# 生成题库
import random
import copy
def generate_sudoku_board():
# 创建一个9x9的二维列表,表示数独棋盘
board = [[0] * 9 for _ in range(9)]
# 递归函数,用于填充数独棋盘的每个单元格
def filling_board(row, col):
# 检查是否填充完成整个数独棋盘
if row == 9:
return True
# 计算下一个单元格的行和列索引
next_row = row if col < 8 else row + 1
next_col = (col + 1) % 9
# 获取当前单元格在小九宫格中的索引
box_row = row // 3
box_col = col // 3
# 随机生成1到9的数字
numbers = random.sample(range(1, 10), 9)
for num in numbers:
# 检查行、列、小九宫格是否已经存在相同的数字
if num not in board[row] and all(board[i][col] != num for i in range(9)) and all(num != board[i][j] for i in range(box_row*3, box_row*3+3) for j in range(box_col*3, box_col*3+3)):
board[row][col] = num
# 递归填充下一个单元格
if filling_board(next_row, next_col):
return True
# 回溯,将当前单元格重置为0
board[row][col] = 0
return False
# 填充数独棋盘
filling_board(0, 0)
return board
def create_board(level): # level数字越大代表游戏难度越大
"""
生成一个随机的数独棋盘,空白格少
"""
board = generate_sudoku_board()
board1 = copy.deepcopy(board)
for i in range(81):
row = i // 9
col = i % 9
if random.randint(0, 9) < level:
board1[row][col] = 0
return (board,board1)
打印一下结果
v = create_board(5)[1]
print(v)
>>>
[[1, 0, 0, 8, 0, 6, 0, 0, 4],
[5, 0, 0, 0, 0, 0, 0, 0, 0],
[0, 6, 0, 7, 0, 2, 0, 0, 1],
[2, 0, 0, 3, 7, 9, 0, 0, 0],
[7, 0, 0, 6, 8, 0, 0, 3, 2],
[0, 0, 5, 4, 0, 0, 7, 6, 9],
[6, 0, 7, 0, 0, 8, 9, 4, 0],
[3, 0, 1, 0, 4, 0, 0, 0, 0],
[9, 0, 4, 5, 6, 0, 0, 2, 7]]
二、界面设计
这里选择使用tkinter
进行界面展示。
希望设计的界面效果如下:
UI部分的代码:
import tkinter as tk
import ctypes
root = tk.Tk()
# 界面优化代码---------------------------------
# 调用api设置成由应用程序缩放
ctypes.windll.shcore.SetProcessDpiAwareness(1)
# 调用api获得当前的缩放因子
ScaleFactor=ctypes.windll.shcore.GetScaleFactorForDevice(0)
# 设置缩放因子
root.tk.call('tk', 'scaling', ScaleFactor/75)
#--------------------------------------------
root.title('数独游戏')
root.geometry('900x1000')
frame = tk.Frame(root,width=500,height=500)
frame.pack()
# 绘制九宫格
def print_board(frame,board):
"""
在界面上显示数独棋盘
"""
for i in range(9):
for j in range(9):
if board[i][j] == 0:
label = tk.Label(frame, text="", width=3, height=2, font=("Arial", 16), bg="white")
else:
label = tk.Label(frame, text=str(board[i][j]), width=3, height=2, font=("Arial", 16), bg="white")
label.grid(row=i, column=j)
def XX(level):
global aa
aa = create_board(level)
return aa
board = XX(5)[1]
print_board(frame,board)
# 添加按键组件
def Button(root,level1,level2):
button1 = tk.Button(root, text="出题:难度1", command=lambda:print_board(frame,XX(level1)[1])) #注意加上lambda!
button1.pack(side=tk.LEFT, padx=10, pady=10)
button2 = tk.Button(root, text="出题:难度2", command=lambda:print_board(frame,XX(level2)[1]))
button2.pack(side=tk.LEFT, padx=10, pady=10)
button3 = tk.Button(root, text="解题",command=lambda:print_board(frame,aa[0]))
button3.pack(side=tk.LEFT, padx=10, pady=10)
button_quit = tk.Button(root,text='退出',command=root.quit)
button_quit.pack(side=tk.RIGHT, padx=10, pady=10)
Button(root,5,7)
root.mainloop()
最终效果:
三、升级优化
以上版本称之为1.0版本,只有出题和解题两个最基本的功能,没有交互的体验感,离一个真正的“数独”游戏还差的很远,下面对代码升级一下,加入用户交互的方式。
- 既然需要交互式设计,自然会想到采用
tk.Entry()
组件。因此绘制九宫格就从之前的文本显示修改为输入框显示,然后将已经出现的数字设置为“可读状态”,避免被修改;而对于待填入的空白格,将输入内容做个限制:只能填入1个数字,而且是1~9,输入其他内容时无效。思路有了,修改代码如下:
# 输入框验证函数
def validate_input(new_value):
if new_value.isdigit() and int(new_value) >= 1 and int(new_value) <= 9:
return True
return False
validate_cmd = frame.register(validate_input)
# 绘制九宫格
def print_board(frame,board):
"""
在界面上显示数独棋盘
"""
for i in range(9):
for j in range(9):
if board[i][j] == 0:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd, "%P"))
else:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd, "%P")) # 注意这里的参数要和上面的label一致,否则会出奇怪的bug~
label.insert(-1,str(board[i][j]))
label.config(state='readonly')
label.grid(row=i, column=j,padx=5,pady=5)
在上面的代码中,Entry组件中的validate
和validatecommand方法
用于设置验证类型和验证函数。validate_input函数
用于验证输入内容,只有当输入内容是1到9的数字时才返回True,否则返回False。root.register方法
用于将这个函数注册为一个验证函数,并返回一个函数名,这个函数名可以用于validatecommand方法
中。%P
表示当前输入框中的内容。
注意:两个entry组件中的参数必须要一致,否则界面就会出现奇怪的bug。😦
看一下效果:
- 上面乍一看没有问题,但是实际上忽略了两个关键问题:
1️⃣ 输入框验证函数需要需要允许空字符情况的存在,否则无法进行删除操作。
2️⃣ 设置成“可读状态”后会对下一步“高亮显示”操作带来困难,需要作另外的处理。因此可以考虑对于已经给出的数字的输入框验证函数保持不变,和上面一样。而待填入的数字的验证函数,添加一个空字符条件判断即可。
更改后代码如下:
# 输入框验证函数 ——针对已给出的数字
def validate_input1(new_value):
if new_value.isdigit() and int(new_value) >= 1 and int(new_value) <= 9:
return True
return False
# 输入框验证函数 ——针对待填入的数字
def validate_input2(new_value):
if new_value == "":
return True
if len(new_value) == 1 and new_value.isdigit():
if 1 <= int(new_value) <= 9:
return True
return False
validate_cmd1 = frame.register(validate_input1)
validate_cmd2 = frame.register(validate_input2)
# 绘制九宫格
def print_board(frame,board):
"""
在界面上显示数独棋盘
"""
for i in range(9):
for j in range(9):
if board[i][j] == 0:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd2, "%P"))
else:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd1, "%P")) # 注意这里的参数要和上面的label一致,否则会出奇怪的bug~
label.insert(-1,str(board[i][j]))
label.grid(row=i, column=j,padx=5,pady=5)
- 填入数字时,界面需要有个反馈,那就是待填入的格子所在的行、列和小九宫格需要高亮显示,并且填入数字之后当前棋盘上所有与之相同的数字也要高亮显示,这样方便玩家进行判定是否正确填入。
通过3个绑定事件来实现这部分功能。
label.bind('<FocusIn>', highlight)
)——聚焦到空白格时进行高亮显示
label.bind('<Leave>', unhighlight)
——焦点离开时进行高亮显示
label.bind('<KeyRelease>', update_board)
——按下键盘时高亮显示相同数字
代码如下:
# 绘制九宫格
def print_board(frame,board):
"""
在界面上显示数独棋盘
"""
for i in range(9):
for j in range(9):
if board[i][j] == 0:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd2, "%P"))
else:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd1, "%P")) # 注意这里的参数要和上面的label一致,否则会出奇怪的bug~
label.insert(-1,str(board[i][j]))
label.grid(row=i, column=j,padx=5,pady=5)
# 添加高亮显示
label.row = i
label.col = j
label.bind('<FocusIn>', highlight)
label.bind('<Leave>', unhighlight)
label.bind('<KeyRelease>', update_board)
# 高亮显示当前格子所在的行、列和小九宫格
def highlight(event):
row, col = event.widget.row, event.widget.col
for i in range(9):
# 高亮行
if i != row:
label = event.widget.master.grid_slaves(row=i, column=col)[0]
label.config(bg='yellow',font=('times',15))
# 高亮列
if i != col:
label = event.widget.master.grid_slaves(row=row, column=i)[0]
label.config(bg='yellow',font=('times',15))
# 高亮小九宫格
r = row // 3 * 3 + i // 3
c = col // 3 * 3 + i % 3
if (r, c) != (row, col):
label = event.widget.master.grid_slaves(row=r, column=c)[0]
label.config(bg='yellow',font=('times',15))
# 高亮当前格子
label = event.widget
label.config(bg='yellow')
# 取消高亮显示
def unhighlight(event):
row, col = event.widget.row, event.widget.col
for i in range(9):
# 取消高亮行
if i != row:
label = event.widget.master.grid_slaves(row=i, column=col)[0]
label.config(bg='white',font=('times',15))
# 取消高亮列
if i != col:
label = event.widget.master.grid_slaves(row=row, column=i)[0]
label.config(bg='white',font=('times',15))
# 取消高亮小九宫格
r = row // 3 * 3 + i // 3
c = col // 3 * 3 + i % 3
if (r, c) != (row, col):
label = event.widget.master.grid_slaves(row=r, column=c)[0]
label.config(bg='white',font=('times',15))
# 取消高亮显示相同数字的格子
for i in range(9):
for j in range(9):
if board[i][j] == board[row][col]:
label = event.widget.master.grid_slaves(row=i, column=j)[0]
label.config(fg='black',font=('times',15))
# 取消高亮当前格子
label = event.widget
label.config(bg='white')
# 更新当前格子的值,并高亮相同数字
def update_board(event):
row, col = event.widget.row, event.widget.col
val = event.widget.get()
if len(val) > 0:
if val.isdigit() and 1 <= int(val) <= 9:
aa[1][row][col] = int(val)
else:
event.widget.delete(0, 'end')
event.widget.insert(0, str(aa[1][row][col]))
# 高亮相同数字的格子
for i in range(9):
for j in range(9):
if aa[1][i][j] == aa[1][row][col]:
label = event.widget.master.grid_slaves(row=i, column=j)[0]
label.config(fg='red',font=('times',15,'bold'))
👁🗨来看一下优化后的效果:
- 最后对整个界面做一些美观的优化,就可以实现一个简单的数独小游戏了。🎮
整个代码:
# 生成题库
import random
import copy
def generate_sudoku_board():
# 创建一个9x9的二维列表,表示数独棋盘
board = [[0] * 9 for _ in range(9)]
# 递归函数,用于填充数独棋盘的每个单元格
def filling_board(row, col):
# 检查是否填充完成整个数独棋盘
if row == 9:
return True
# 计算下一个单元格的行和列索引
next_row = row if col < 8 else row + 1
next_col = (col + 1) % 9
# 获取当前单元格在小九宫格中的索引
box_row = row // 3
box_col = col // 3
# 随机生成1到9的数字
numbers = random.sample(range(1, 10), 9)
for num in numbers:
# 检查行、列、小九宫格是否已经存在相同的数字
if num not in board[row] and all(board[i][col] != num for i in range(9)) and all(num != board[i][j] for i in range(box_row*3, box_row*3+3) for j in range(box_col*3, box_col*3+3)):
board[row][col] = num
# 递归填充下一个单元格
if filling_board(next_row, next_col):
return True
# 回溯,将当前单元格重置为0
board[row][col] = 0
return False
# 填充数独棋盘
filling_board(0, 0)
return board
def create_board(level): # level数字越大代表游戏难度越大
"""
生成一个随机的数独棋盘,空白格少
"""
board = generate_sudoku_board()
board1 = copy.deepcopy(board)
for i in range(81):
row = i // 9
col = i % 9
if random.randint(0, 9) < level:
board1[row][col] = 0
return (board,board1)
import tkinter as tk
import ctypes
from tkinter import messagebox
root = tk.Tk()
# 界面优化代码---------------------------------
# 调用api设置成由应用程序缩放
ctypes.windll.shcore.SetProcessDpiAwareness(1)
# 调用api获得当前的缩放因子
ScaleFactor=ctypes.windll.shcore.GetScaleFactorForDevice(0)
# 设置缩放因子
root.tk.call('tk', 'scaling', ScaleFactor/75)
#--------------------------------------------
root.title('数独游戏')
root.geometry('900x1000')
frame = tk.Frame(root,width=500,height=500)
frame.pack()
# 输入框验证函数 ——针对已给出的数字
def validate_input1(new_value):
if new_value.isdigit() and int(new_value) >= 1 and int(new_value) <= 9:
return True
return False
# 输入框验证函数 ——针对待填入的数字
def validate_input2(new_value):
if new_value == "":
return True
if len(new_value) == 1 and new_value.isdigit():
if 1 <= int(new_value) <= 9:
return True
return False
validate_cmd1 = frame.register(validate_input1)
validate_cmd2 = frame.register(validate_input2)
# 绘制九宫格
def print_board(frame,board):
"""
在界面上显示数独棋盘
"""
for i in range(9):
for j in range(9):
if board[i][j] == 0:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd2, "%P"))
else:
label = tk.Entry(frame,width=4,font=('TimesNewom',15,'bold'), validate="key", validatecommand=(validate_cmd1, "%P")) # 注意这里的参数要和上面的label一致,否则会出奇怪的bug~
label.insert(-1,str(board[i][j]))
label.grid(row=i, column=j,padx=10,pady=10)
# 添加高亮显示
label.row = i
label.col = j
label.bind('<FocusIn>', highlight)
label.bind('<Leave>', unhighlight)
label.bind('<KeyRelease>', update_board)
# 高亮显示当前格子所在的行、列和小九宫格
def highlight(event):
row, col = event.widget.row, event.widget.col
for i in range(9):
# 高亮行
if i != row:
label = event.widget.master.grid_slaves(row=i, column=col)[0]
label.config(bg='yellow',font=('times',15))
# 高亮列
if i != col:
label = event.widget.master.grid_slaves(row=row, column=i)[0]
label.config(bg='yellow',font=('times',15))
# 高亮小九宫格
r = row // 3 * 3 + i // 3
c = col // 3 * 3 + i % 3
if (r, c) != (row, col):
label = event.widget.master.grid_slaves(row=r, column=c)[0]
label.config(bg='yellow',font=('times',15))
# 高亮当前格子
label = event.widget
label.config(bg='yellow')
# 取消高亮显示
def unhighlight(event):
row, col = event.widget.row, event.widget.col
for i in range(9):
# 取消高亮行
if i != row:
label = event.widget.master.grid_slaves(row=i, column=col)[0]
label.config(bg='white',font=('times',15))
# 取消高亮列
if i != col:
label = event.widget.master.grid_slaves(row=row, column=i)[0]
label.config(bg='white',font=('times',15))
# 取消高亮小九宫格
r = row // 3 * 3 + i // 3
c = col // 3 * 3 + i % 3
if (r, c) != (row, col):
label = event.widget.master.grid_slaves(row=r, column=c)[0]
label.config(bg='white',font=('times',15))
# 取消高亮显示相同数字的格子
for i in range(9):
for j in range(9):
if board[i][j] == board[row][col]:
label = event.widget.master.grid_slaves(row=i, column=j)[0]
label.config(fg='black',font=('times',15))
# 取消高亮当前格子
label = event.widget
label.config(bg='white')
# 更新当前格子的值,并高亮相同数字
def update_board(event):
row, col = event.widget.row, event.widget.col
val = event.widget.get()
if len(val) > 0:
if val.isdigit() and 1 <= int(val) <= 9:
aa[1][row][col] = int(val)
else:
event.widget.delete(0, 'end')
event.widget.insert(0, str(aa[1][row][col]))
# 高亮相同数字的格子
for i in range(9):
for j in range(9):
if aa[1][i][j] == aa[1][row][col]:
label = event.widget.master.grid_slaves(row=i, column=j)[0]
label.config(fg='red',font=('times',15,'bold'))
def XX(level):
global aa # 通过全局变量可以获取每次按键刷新后的棋盘
aa = create_board(level)
return aa
board = XX(5)[1]
print_board(frame,board)
# 添加按键组件
def Button(root,level1,level2):
button1 = tk.Button(root, text="出题:难度1", command=lambda:print_board(frame,XX(level1)[1]), relief="raised",height=2,font=('楷体',10,'bold')) #注意加上lambda!
button1.pack(side=tk.LEFT, padx=10, pady=10)
button2 = tk.Button(root, text="出题:难度2", command=lambda:print_board(frame,XX(level2)[1]), relief="raised",height=2,font=('楷体',10,'bold'))
button2.pack(side=tk.LEFT, padx=10, pady=10)
button3 = tk.Button(root, text="解题",command=lambda:print_board(frame,aa[0]), relief="raised",height=2,font=('楷体',10,'bold'))
button3.pack(side=tk.LEFT, padx=10, pady=10)
button4 = tk.Button(root,text="验证",command = lambda:check(aa[1]), relief="raised",height=2,font=('楷体',10,'bold'))
button4.pack(side=tk.LEFT, padx=10, pady=10)
button_quit = tk.Button(root,text='退出',command=root.quit,bg='red', relief="raised",height=2,font=('楷体',10,'bold'))
button_quit.pack(side=tk.RIGHT, padx=10, pady=10)
Button(root,5,7)
# 判断输赢
def check_victory(sudo):
# 检查每行是否有重复数字
for row in sudo:
if len(set(row)) != 9:
return False
# 检查每列是否有重复数字
for col in range(9):
column = [sudo[row][col] for row in range(9)]
if len(set(column)) != 9:
return False
# 检查每个3x3的小九宫格内是否有重复数字
for box_row in range(3):
for box_col in range(3):
box = [sudo[box_row*3 + i][box_col*3 + j] for i in range(3) for j in range(3)]
if len(set(box)) != 9:
return False
return True
def check(sudo):
if sudo == aa[1]:
if check_victory(sudo) :
messagebox.showinfo("提示", "恭喜你,你赢了!")
else:
messagebox.showinfo("提示", "很遗憾,你输了!")
root.mainloop()
最终效果:
总结
- 这里将每个功能用函数的形式去实现,而如果整个代码写成类的形式封装在一起会更好。
- 鼠标在聚焦输入框和离开输入框时存在一些问题,界面会稍微浮动变化。
- 不足之处,欢迎指正!🔍