大家好,欢迎来到IT知识分享网。
作品展示:
16宫格里面的4*4小格子可以凑满1-16,旁边的7宫格格2份
背景需求:
做完了1-20宫格的A4模板,最后做一个16宫格小格子(附加7宫格2套)的样式,只有4宫格(2*2)、9宫格(3*3)和16宫格(4*4)三款适用小格子统计
【教学类-43-23】 数独(五)16宫格(4*4可算全部数字)无空行A4模板 左右结构(附加7宫格)
素材准备
代码展示:
''' 目的:数独26 16宫格一页1份连在一起小格子1-16,右侧方多余空格做7宫格2套 作者:阿夏(参考) 时间:2024年1月16日 13:35 ''' import random from win32com.client import constants,gencache from win32com.client.gencache import EnsureDispatch from win32com.client import constants # 导入枚举常数模块 import os,time import docx from docx import Document from docx.shared import Pt from docx.shared import RGBColor from docx.enum.text import WD_PARAGRAPH_ALIGNMENT from docx.oxml.ns import qn from docxtpl import DocxTemplate import pandas as pd from docx2pdf import convert from docx.shared import RGBColor # 生成题库 import random import copy num= int(input('生成几份\n')) kk1=int(input('16宫格空格数量,输入3,就是30%,就是空30%)\n')) kk2=int(input('7宫格空格数量,输入5,就是50%,就是空一半)\n')) # 制作"单元格"# 几宫格 ll=['1607',] mm=['1112'] # 11对应的套数是2套题目 11宫格可以插入4宫格5个 nn=['2828',]# 3对应的写入单元格的数字大小36磅 # 新建一个”装N份word和PDF“的临时文件夹 imagePath1=r'C:\Users\jg2yXRZ\OneDrive\桌面\数独\零时Word' if not os.path.exists(imagePath1): # 判断存放图片的文件夹是否存在 os.makedirs(imagePath1) # 若图片文件夹不存在就创建 bg=[] bgszm=[] P=[] PP=[] for m in range(len(ll)): # 0-2 bg.clear() PP.clear() for n in range(int(len(ll[m])/2)): # 0-4 # word名称:只调用第一个宫格数字11、12 name=int(ll[m][0:2]) # word长宽:只调用第一个宫格数字11的长2 long=int(mm[m][0:1]) # word长宽:只调用第一个宫格数字11的宽1 wide=int(mm[m][1:2]) # 11-15主要宫格 4-5补充宫格 hs=int(ll[m][n*2:n*2+2]) # print(hs) # 小宫格的起始点坐标 【1104】 h=int(ll[m][0:2]) # print(h),只要11 # print(h),只要11 # 底边几套.底边看有2份 db=int(mm[m][n*2:n*2+2][0]) # print(db) # # 侧边看有1份 cb=int(mm[m][n*2:n*2+2][1]) # print(cb) size=int(nn[m][n*2:n*2+2]) # print(size) # 写入单元格数字的大小(撑满格子) # 计算不同模板中的单元格坐标,放在bg里 # 棋盘格子数量, # 如果长方形:底边3*侧边2,3*2=6套 ,就是10*10宫格,底边格子数量就是10*3=30,侧边格子数量就是10*2=20, db_size = hs*db # print(db_size) cb_size= hs*cb # print(cb_size) # print('{}宫格排列底{}侧{}共{}套,底边格子数{}'.format(hs,db,cb,db*cb,db_size )) # print('{}宫格排列底{}侧{}共{}套,侧边格子数{}'.format(hs,db,cb,db*cb,cb_size )) bgszm=[] if hs >15: # 确定11-15宫格的左上角坐标 00 04 40 44 中间没有分栏的空格了,所以hs不用+1 for a in range(0,cb_size,hs): # 0-11每隔4,写一个坐标 侧边y for b in range(0,db_size,hs): # 0-11每隔4,写一个坐标 侧边x bgszm.append('{}{}'.format('%02d'%a,'%02d'%b)) # for b in bgszm: # bgbg.append(b) if hs <=15: for a in range(0,h,hs): # 0-11每隔4,写一个坐标 侧边y for b in range(h,h+db_size,h): # 0-11每隔4,写一个坐标 侧边x bgszm.append('{}{}'.format('%02d'%a,'%02d'%b)) # ['0020', '0920', '1820'] bgszm=bgszm[0:2] # 只要前2个 print(bgszm) # 算出基础坐标 11宫格、4宫格 # 转为元祖 start_coordinates = [(int(s[0:2]), int(s[2:4])) for s in bgszm] cell_coordinates = [] # 推算每个起始格子后面的单元格数字 for start_coord in start_coordinates: i, j = start_coord subgrid_coordinates = [] for x in range(hs): for y in range(hs): subgrid_coordinates.append((i + x, j + y)) cell_coordinates.append(subgrid_coordinates) # 打印结果(元祖样式) bgbg=[] for coordinates in cell_coordinates: # print(coordinates) # [(4, 8), (4, 9), (4, 10), (5, 8), (5, 9), (5, 10), (6, 8), (6, 9), (6, 10)] for c in coordinates: # print(c) # 元组 (1, 2) 样式 s = ''.join(str(num).zfill(2) for num in c) # zfill将元组 (1, 2) 转换为字符串 '0102' 特别是(5,10)这种必须转成2个数字0510 # print(str(s)) # '12' bgbg.append(s) # '0102' # print(bg) # # ['0000', '0001', '0002', '0003', '0100', '0101', '0102', '0103', '0200', '0201', '0202', '0203', '0300', '0301', '0302', '0303', '0012', '0013', '0014', '0015', '0112', '0113', '0114', '0115', '0212', '0213', '0214', '0215', '0312', '0313', '0314', '0315', '1200', '1201', '1202', '1203', '1300', '1301', '1302', '1303', '1400', '1401', '1402', '1403', '1500', '1501', '1502', '1503', '1204', '1205', '1206', '1207', '1304', '1305', '1306', '1307', '1404', '1405', '1406', '1407', '1504', '1505', '1506', '1507', '1208', '1209', '1210', '1211', '1308', '1309', '1310', '1311', '1408', '1409', '1410', '1411', '1508', '1509', '1510', '1511', '1212', '1213', '1214', '1215', '1312', '1313', '1314', '1315', '1412', '1413', '1414', '1415', '1512', '1513', '1514', '1515', '1216', '1217', '1218', '1219', '1316', '1317', '1318', '1319', '1416', '1417', '1418', '1419', '1516', '1517', '1518', '1519', '1220', '1221', '1222', '1223', '1320', '1321', '1322', '1323', '1420', '1421', '1422', '1423', '1520', '1521', '1522', '1523'] # print(len(bg)) for b in bgbg: bg.append(b) # 生成PDf for z in range(num): for j in range(db*cb): # 3宫格,4*3=12套 # ————————————————生成随机宫格,按比例空缺格子 # 版权声明:本文为CSDN博主「Vaeeeeeee」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 # 原文链接:https://blog.csdn.net/m0_/article/details/ def generate_sudoku_board(): # 函数体生成数独库表盘 # 创建一个9x9的二维列表,表示数独棋盘 board = [[0] * hs for _ in range(hs)] # 递归函数,用于填充数独棋盘的每个单元格 def filling_board(row, col): # 检查是否填充完成整个数独棋盘 if row == hs: return True # 计算下一个单元格的行和列索引 next_row = row if col < hs-1 else row + 1 next_col = (col + 1) % hs import math if hs > 10: r = int(math.sqrt(hs)) print(r) # 获取当前单元格在小九宫格中的索引 box_row = row // r box_col = col // r # 随机生成1到9的数字 numbers = random.sample(range(1, hs+1), hs) for num in numbers: # 检查行、列、小九宫格是否已经存在相同的数字 if num not in board[row] and all(board[i][col] != num for i in range(hs)) and all(num != board[i][j] for i in range(box_row*r, box_row*r+r) for j in range(box_col*r, box_col*r+r)): board[row][col] = num # 递归填充下一个单元格 if filling_board(next_row, next_col): return True # 回溯,将当前单元格重置为0 board[row][col] = 0 else: # 获取当前单元格在小九宫格中的索引 box_row = row box_col = col # 随机生成1到9的数字 numbers = random.sample(range(1, hs+1), hs) for num in numbers: # 检查行、列、小九宫格是否已经存在相同的数字 if num not in board[row] and all(board[i][col] != num for i in range(hs)) and all(num != board[i][j] for i in range(box_row, box_row) for j in range(box_col, box_col)): 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数字越大代表游戏难度越大 """ 生成一个随机的数独棋盘,空白格少 """ board = generate_sudoku_board() board1 = copy.deepcopy(board) # 11-15宫格不好做,建议比例30% if hs>10: blanks = random.sample(range(hs*hs), int(hs*hs*kk1/10)) # 4-5宫格容易,建议比例50% else: blanks = random.sample(range(hs*hs), int(hs*hs*kk2/10)) for i in blanks: row = i // hs col = i % hs board[row][col] = 0 # if random.randint(0, hs) < level: # board1[row][col] = 0 return board # 定量出现空白格子 v = create_board() # 81空34、46 for a1 in v: for a2 in a1: if a2==0: # 如果某个元素==0,就替换成空 PP.append('') else: # 如果某个元素非0,就写入本身的数字 PP.append(a2) P=PP # print(P) # print(len(P)) # print(bg) # # # 11宫格 322格子 12宫格 384格子 # print(len(bg)) doc = Document(r'C:\Users\jg2yXRZ\OneDrive\桌面\数独\数独长方形({}宫格)合并版.docx'.format(name)) # doc = Document(r'C:\Users\jg2yXRZ\OneDrive\桌面\数独\数独长方形(15宫格)合并版.docx') # table = doc.tables[0] # 表0,表2 写标题用的 # 标题写入3、5单元格 for t in range(0,len(bg)): # 0-5是最下面一行,用来写卡片数字 pp=int(bg[t][0:2]) # =int(bg[t][2:4]) k=str(P[t]) # 提取list图案列表里面每个图形 t=索引数字 print(pp,,k) # 图案符号的字体、大小参数 run=table.cell(pp,).paragraphs[0].add_run(k) # 在单元格0,0(第1行第1列)输入第0个图图案 run.font.name = '黑体'#输入时默认华文彩云字体 # run.font.size = Pt(46) #输入字体大小默认30号 换行(一页一份大卡片 run.font.size = Pt(size) #是否加粗 # run.font.color.rgb = RGBColor(150,150,150) #数字小,颜色深0-255 run.font.color.rgb = RGBColor(50,50,50) #数字小,颜色深0-255 run.bold=True # paragraph.paragraph_format.line_spacing = Pt(180) #数字段间距 r = run._element r.rPr.rFonts.set(qn('w:eastAsia'), '黑体')#将输入语句中的中文部分字体变为华文行楷 table.cell(pp,).paragraphs[0].alignment = WD_PARAGRAPH_ALIGNMENT.CENTER#居中 doc.save(r'C:\Users\jg2yXRZ\OneDrive\桌面\数独\零时Word\{}.docx'.format('%02d'%(z+1)))#保存为XX学号的电话号码word time.sleep(2) from docx2pdf import convert # docx 文件另存为PDF文件 inputFile = r"C:/Users/jg2yXRZ/OneDrive/桌面/数独/零时Word/{}.docx".format('%02d'%(z+1))# 要转换的文件:已存在 outputFile = r"C:/Users/jg2yXRZ/OneDrive/桌面/数独/零时Word/{}.pdf".format('%02d'%(z+1)) # 要生成的文件:不存在 # 先创建 不存在的 文件 f1 = open(outputFile, 'w') f1.close() # 再转换往PDF中写入内容 convert(inputFile, outputFile) print('----------第4步:把都有PDF合并为一个打印用PDF------------') # 多个PDF合并(CSDN博主「红色小小螃蟹」,https://blog.csdn.net/yangcunbiao/article/details/) import os from PyPDF2 import PdfMerger target_path = 'C:/Users/jg2yXRZ/OneDrive/桌面/数独/零时Word' pdf_lst = [f for f in os.listdir(target_path) if f.endswith('.pdf')] pdf_lst = [os.path.join(target_path, filename) for filename in pdf_lst] pdf_lst.sort() file_merger = PdfMerger() for pdf in pdf_lst: print(pdf) file_merger.append(pdf) file_merger.write("C:/Users/jg2yXRZ/OneDrive/桌面/数独/(打印合集)05长方形数独宫格25.0 {}宫格小格子无空附加{}乘{}等于{}套({}人{}份).pdf" .format(name,long,wide,long*wide,num,num)) # file_merger.write("C:/Users/jg2yXRZ/OneDrive/桌面/数独/(打印合集)05长方形数独宫格25.0 .pdf") file_merger.close() # doc.Close() # # print('----------第5步:删除临时文件夹------------') import shutil shutil.rmtree('C:/Users/jg2yXRZ/OneDrive/桌面/数独/零时Word') #递归删除文件夹,即:删除非空文件夹 time.sleep(3) # 防止转换时报错,预留生成时间
重点说明:
本代码用两种生成数字的方法,一种是横纵+平方根代码,一种是只考虑横纵的代码,终于搞清楚两者的差别e
终端输入
结论:
1-20内的A4大小宫格数独都做完了。宫格数独系列完结
免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://haidsoft.com/155514.html