作品展示:
背景需求:
把以下四款3连题 混在一起,每种题目随机抽取11题,一共44格
出现问题:
1、+- 、-+里面有重复题
2、升序排列最好竖排展示
素材准备:
问题改正
1、单元格修改:确保竖列写入
修改前
修改后
2、少了一部分的去重排序
修改前
修改后
代码展示
'''
X-Y 3连加减 单元格竖排 4类题型等比例抽取。44格子每种11题
1、按比例抽题:44格子每种11题,3、3++ 3-- 3+- 3-+ 比例各2.5时间:2024年1月10日 10:46
作者:阿夏
'''import random
from win32com.client import constants,gencache
from win32com.client.gencache import EnsureDispatch
from win32com.client import constants # 导入枚举常数模块
import os,timeimport 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 qnfrom docxtpl import DocxTemplate
import pandas as pd
from docx2pdf import convert
from docx.shared import RGBColor# 第一步:制作不重复所有“+-”、不重复所有减法# 不重复的数字题
num=int(input('打印几份(必须是双数)\n'))# int(input('一共几个单元格(55个)\n'))
# classroom=input('班级(输入中、大)\n')
# 本次是2.5浮点数,不是整数
bl=float(input('抽取比例,4类题2.5\n'))
size=20
height1=12
weight1=4gz=(height1-1)*weight1
sum1=int(input('X-Y以内的“三连题” 最小数字X\n'))
sum2=int(input('X-Y以内的“三连题” 最大数字Y\n'))
l=int(input('输入1,题目打乱,输入2,++ -- +- -+\n'))# for sum in [sum2]:
# 5以内“+-”题共21题P=[]
jiajia=[]
# 3加加
for a in range(0,sum2+1): # 起始数字就是10,就是排除掉0-10之间的数字for b in range(0,sum2+1): # 起始数字为0,for c in range(0,sum2+1): # 起始数字就是10,就是排除掉0-10之间的数字if sum1<=a+b+c<sum2+1 or sum1<=a+c+b<sum2+1 or sum1<=b+a+c<sum2+1 or \sum1<=b+c+a<sum2+1 or sum1<=c+a+b<sum2+1 or sum1<=c+b+a<sum2+1 : # print('{}+{}='.format(a,b))jiajia.append('{}+{}+{}='.format(a,b,c))jiajia.append('{}+{}+{}='.format(a,c,b))jiajia.append('{}+{}+{}='.format(b,a,c))jiajia.append('{}+{}+{}='.format(b,c,a))jiajia.append('{}+{}+{}='.format(c,a,b))jiajia.append('{}+{}+{}='.format(c,b,a))# 0-5 三连加 56道# 0-10 三连加 286道else:pass
jiajia=list(set(jiajia)) # 加法题 去重 42变成21题
jiajia.sort() # 升序
print(len(jiajia))# 3减减
jianjian=[]
for a in range(0,sum2+1): # 起始数字就是10,就是排除掉0-10之间的数字for b in range(0,sum2+1): # 起始数字为0,for c in range(0,sum2+1): # 起始数字就是10,就是排除掉0-10之间的数字if sum1<=a-b-c<sum2+1 and a>=b and a>=c:jianjian.append('{}-{}-{}='.format(a,b,c))if sum1<=a-c-b<sum2+1 and a>=c and a>=b:jianjian.append('{}-{}-{}='.format(a,c,b))if sum1<=b-a-c<sum2+1 and b>=a and b>=c:jianjian.append('{}-{}-{}='.format(b,a,c))if sum1<=b-c-a<sum2+1 and b>=c and b>=a:jianjian.append('{}-{}-{}='.format(b,c,a))if sum1<=c-a-b<sum2+1 and c>=a and c>=b:jianjian.append('{}-{}-{}='.format(c,a,b))if sum1<=c-b-a<sum2+1 and c>=b and c>=a: jianjian.append('{}-{}-{}='.format(c,b,a)) # 0-5 三连减 34道# 0-10 三连减 161题 else:pass
jianjian=list(set(jianjian)) # 减法题 去重 42变成21题
jianjian.sort() # 升序
print(len(jianjian)) # 21# 3加减
jiajian=[]
for a in range(0,sum2+1): # for b in range(0,sum2+1): # 起始数字为0,for c in range(0,sum2+1): # # 三个数字先加后减最后答案在0-5之间,三个数字本身加起来是0-5之间if sum1<=a+b-c<sum2+1 and a+b>=c and sum1<a+b+c<sum2+1: jiajian.append('{}+{}-{}='.format(a,b,c))if sum1<=a+c-b<sum2+1 and a+c>=b and sum1<=a+b+c<sum2+1: jiajian.append('{}+{}-{}='.format(a,c,b))if sum1<=b+a-c<sum2+1 and b+a>=c and sum1<=a+b+c<sum2+1: jiajian.append('{}+{}-{}='.format(b,a,c))if sum1<=b+c-a<sum2+1 and b+c>=a and sum1<=a+b+c<sum2+1: jiajian.append('{}+{}-{}='.format(b,c,a))if sum1<=c+a-b<sum2+1 and c+a>=b and sum1<=a+b+c<sum2+1: jiajian.append('{}+{}-{}='.format(c,a,b))if sum1<=c+b-a<sum2+1 and c+b>=a and sum1<=a+b+c<sum2+1: jiajian.append('{}+{}-{}='.format(c,b,a))# 0-5 三连加减减 42道# 0-10 三连加减减 216道else:pass
jiajian=list(set(jiajian)) # 减法题 去重 42变成21题
jiajian.sort() # 升序
print(len(jiajian)) # 21# 3减加
jianjia=[]
for a in range(0,sum2+1): # for b in range(0,sum2+1): # 起始数字为0,for c in range(0,sum2+1): # # 三个数字先加后减最后答案在0-5之间,三个数字本身加起来是0-5之间# 先减后加,确保第一个数大于第二个数if sum1<=a-b+c<sum2+1 and a>=b and sum1<a+b+c<sum2+1: jianjia.append('{}-{}+{}='.format(a,b,c))if sum1<=a-c+b<sum2+1 and a>=c and sum1<=a+b+c<sum2+1: jianjia.append('{}-{}+{}='.format(a,c,b))if sum1<=b-a+c<sum2+1 and b>=a and sum1<=a+b+c<sum2+1: jianjia.append('{}-{}+{}='.format(b,a,c))if sum1<=b-c+a<sum2+1 and b>=c and sum1<=a+b+c<sum2+1: jianjia.append('{}-{}+{}='.format(b,c,a))if sum1<=c-a+b<sum2+1 and c>=a and sum1<=a+b+c<sum2+1: jianjia.append('{}-{}+{}='.format(c,a,b))if sum1<=c-b+a<sum2+1 and c>=b and sum1<=a+b+c<sum2+1: jianjia.append('{}-{}+{}='.format(c,b,a))# 0-5 三连减加 34道# 0-10 三连减加 161道else:pass
jianjia=list(set(jianjia)) # 减法题 去重 42变成21题
jianjia.sort() # 升序
print(len(jianjia)) # 21P=len(jianjian)+len(jiajia)+len(jiajian)+len(jianjia)
print(P)# 0-5加法减法题目总数42L=jiajia+jianjian+jiajian+jianjia
print(L)
print(len(L))
# ['0+0=', '0+1=', '0+2=', '0+3=', '0+4=', '0+5=', '1+0=', '1+1=', '1+2=', '1+3=', '1+4=', '2+0=', '2+1=', '2+2=', '2+3=', '3+0=', '3+1=', '3+2=', '4+0=', '4+1=', '5+0=', '0-0=', '1-0=', '1-1=', '2-0=', '2-1=', '2-2=', '3-0=', '3-1=', '3-2=', '3-3=', '4-0=', '4-1=', '4-2=', '4-3=', '4-4=', '5-0=', '5-1=', '5-2=', '5-3=', '5-4=', '5-5=']# 第一行的班级和项目
A=[]
# c='{}'.format(classroom)if P>gz: # 0-10等于132题,大于55,单元格数量55print('数学题总数大于55,实际题目数量{}'.format(gz))sl=Ptl1=int(float(gz*bl*10/100)) # 加法题的题量是 21*50/100 可能是浮点数10.5,所以要用int=10# print(tl1)tl3=tl2=tl1tl4=gz-(tl3+tl2+tl1)print(tl4)print(tl3) print(tl2)print(tl1)title='{}-{}“3连题4类”{}抽{}题1/4'.format(sum1,sum2,P,gz)if P<=gz: # 0-5等于42题,小于于55,单元格数量42print('数学题总数小于55,实际题目数量{}'.format(P))sl=Ptl1=len(jiajia) # 加法题的题量是 21*50/100 可能是浮点数10.5,所以要用int=11print(tl1)tl3=tl2=tl1tl4=gz-(tl3+tl2+tl1)title='{}-{}“+-”{}抽{}题 全部包含'.format(sum1,sum2,P,P)print(tl4)print(tl3)d=['0002']
# 表格0 表格2的 03 05单元格里写入标题信息c
# A.append(c)
A.append(title)
print(A) # 制作"单元格"
bgall=[]
for bb in d:bgall.append(bb)for y in range(0,weight1): # 4for x in range(1,height1): # 12s1='{}{}'.format('%02d'%x,'%02d'%y) # 数字加空格bgall.append(s1)
print(bgall)
print(len(bgall))# 不同情况下的单元格数量
if P <=gz:bg=bgall[0:2+P]print(bg)print(len(bg))
else:bg=bgall[0:2+gz]print(bg)print(len(bg))# ['0003', '0005', '0100', '0101', '0102', '0103', '0104', '0200', '0201', '0202', '0203', '0204', '0300', '0301', '0302', '0303', '0304', '0400', '0401', '0402', '0403', '0404', '0500']# 新建一个”装N份word和PDF“的临时文件夹
imagePath1=r'C:\Users\jg2yXRZ\OneDrive\桌面\加减法\零时Word'
if not os.path.exists(imagePath1): # 判断存放图片的文件夹是否存在os.makedirs(imagePath1) # 若图片文件夹不存在就创建D=[]
n=int(num/2)
for z in range(0,n): #多少份 # 标题说明# 新建worddoc = Document(r'C:\Users\jg2yXRZ\OneDrive\桌面\加减法\05三连加减一页两份.docx') for j in range(2):D.clear()if P <=gz:# D=[]# 小于9的题目,要计算一共有几题,写入等量的单元格内, C1=random.sample(jiajia,tl1) for cc1 in C1:D.append(cc1)C2=random.sample(jianjian,tl2) for cc2 in C2:D.append(cc2)C3=random.sample(jiajian,tl3) for cc3 in C3:D.append(cc3)C4=random.sample(jianjia,tl4) for cc4 in C4:D.append(cc4)if l==1:random.shuffle(D) # 如果=1,加减混合打乱if l==2:# 如果=2,先出加法,再出减法passprint(D)print(len(D))D.insert(0,title) # 写入班级,项目名称# D.insert(0,classroom) else:# D=[]# 大于9的题目,只要抽取55题,多余的写不下,C1=random.sample(jiajia,tl1) for cc1 in C1:D.append(cc1)C2=random.sample(jianjian,tl2) for cc2 in C2:D.append(cc2) C3=random.sample(jiajian,tl3) for cc3 in C3:D.append(cc3) C4=random.sample(jianjia,tl4) for cc4 in C4:D.append(cc4)if l==1:random.shuffle(D) # 如果=1,加减混合打乱if l==2:# 如果=2,先出加法,再出减法passprint(D)print(len(D))D.insert(0,title) # 写入班级,项目名称# D.insert(0,classroom) # # 房间模板(第一个表格)要写入的门牌号列表 table = doc.tables[j] # 表0,表2 写标题用的# 标题写入3、5单元格 for t in range(0,len(bg)): # 0-5是最下面一行,用来写卡片数字pp=int(bg[t][0:2]) # qq=int(bg[t][2:4])k=str(D[t]) # 提取list图案列表里面每个图形 t=索引数字print(pp,qq,k)# 图案符号的字体、大小参数run=table.cell(pp,qq).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-255run.font.color.rgb = RGBColor(150,150,150) #数字小,颜色深0-255run.bold=True# paragraph.paragraph_format.line_spacing = Pt(180) #数字段间距r = run._elementr.rPr.rFonts.set(qn('w:eastAsia'), '黑体')#将输入语句中的中文部分字体变为华文行楷table.cell(pp,qq).paragraphs[0].alignment = WD_PARAGRAPH_ALIGNMENT.LEFT #居中
# 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/125248205)
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)if P <=gz :if l==1: # 打乱file_merger.write("C:/Users/jg2yXRZ/OneDrive/桌面/加减法/(打印合集)03“三连题”(一页两份 ){}题{}-{}之间“4类三连题”数量{}比{}比{}比{}乱序(共{}题抽{}题)({}共{}人打印{}张).pdf" .format(gz,'%02d'%sum1,'%02d'%sum2,bl,bl,bl,10-bl*3,'%03d'%P,'%02d'%P,c,num,n))else: # 先加后减file_merger.write("C:/Users/jg2yXRZ/OneDrive/桌面/加减法/(打印合集)03“三连题”(一页两份 ){}题{}-{}之间“4类三连题”数量{}比{}比{}比{}顺序(共{}题抽{}题)({}共{}人打印{}张).pdf" .format(gz,'%02d'%sum1,'%02d'%sum2,bl,bl,bl,10-bl*3,'%03d'%P,'%02d'%P,c,num,n))
else:if l==1: # # 打乱file_merger.write("C:/Users/jg2yXRZ/OneDrive/桌面/加减法/(打印合集)03“三连题”(一页两份 ){}题{}-{}之间“4类三连题”数量{}比{}比{}比{}乱序(共{}题抽{}题)({}共{}人打印{}张).pdf".format(gz,'%02d'%sum1,'%02d'%sum2,bl,bl,bl,10-bl*3,'%03d'%P,gz,c,num,n))else: # 先加后减file_merger.write("C:/Users/jg2yXRZ/OneDrive/桌面/加减法/(打印合集)03“三连题”(一页两份 ){}题{}-{}之间“4类三连题”数量{}比{}比{}比{}顺序(共{}题抽{}题)({}共{}人打印{}张).pdf".format(gz,'%02d'%sum1,'%02d'%sum2,bl,bl,bl,10-bl*3,'%03d'%P,gz,c,num,n))file_merger.close()
# doc.Close()# # print('----------第5步:删除临时文件夹------------')
import shutil
shutil.rmtree('C:/Users/jg2yXRZ/OneDrive/桌面/加减法/零时Word') #递归删除文件夹,即:删除非空文件夹
终端输入
结果展示:
0-5 3连题4类 一共抽取44题,并按照顺序排列(同类题目在一起)
1、++题 11题
2、--题 11题
3、+-题 11题
4、-+题 11题
0-10 3连题4类 一共抽取44题,并按照顺序排列(同类题目在一起)
1、++题 11题
2、--题 11题
3、+-题 11题
4、-+题 11题
第二种:乱序排列
感悟:
生成的数学题一定先老师自己做一下,验证是否正确。,