作品展示:
背景需求:
把以下四款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,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
# 第一步:制作不重复所有“+-”、不重复所有减法
# 不重复的数字题
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=4
gz=(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)) # 21
P=len(jianjian)+len(jiajia)+len(jiajian)+len(jianjia)
print(P)# 0-5加法减法题目总数42
L=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,单元格数量55
print('数学题总数大于55,实际题目数量{}'.format(gz))
sl=P
tl1=int(float(gz*bl*10/100)) # 加法题的题量是 21*50/100 可能是浮点数10.5,所以要用int=10
# print(tl1)
tl3=tl2=tl1
tl4=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,单元格数量42
print('数学题总数小于55,实际题目数量{}'.format(P))
sl=P
tl1=len(jiajia) # 加法题的题量是 21*50/100 可能是浮点数10.5,所以要用int=11
print(tl1)
tl3=tl2=tl1
tl4=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): # 4
for x in range(1,height1): # 12
s1='{}{}'.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): #多少份
# 标题说明
# 新建word
doc = 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,先出加法,再出减法
pass
print(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,先出加法,再出减法
pass
print(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-255
run.font.color.rgb = RGBColor(150,150,150) #数字小,颜色深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,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题
第二种:乱序排列
感悟:
生成的数学题一定先老师自己做一下,验证是否正确。,