
 
背景需求
 
4月23日听了一个MJB的征文培训,需要写会议记录
 

 
把资料黏贴到模版后,发现每行需要有画满下划线
 

 

 
 原来做这套资料,就是手动按空格到一行末,有空格才会出现下划线,也就是要按很多的空格(凑满一行)
 

 
或者一行行的复制复制,因为文字有长短,所以后面的空格也有长有短
 

 
需求:
 
Python批量自动给最后添加适合的空格数,自动实现没有文字,但有空格,有下划线
 

 
第1问
 

 
 
代码展示
 
from docx import Documentdef add_solid_underline_with_spaces(doc_path, output_path, target_length=29):"""为Word文档每段添加实线下划线,并用全角空格填充至指定长度(仅对每段最后一行末尾添加空格和下划线)参数:doc_path (str): 输入文档路径output_path (str): 输出文档路径target_length (int): 目标字符宽度(中文算2,英文算1)"""# 加载文档doc = Document(doc_path)for paragraph in doc.paragraphs:if paragraph.text.strip():  # 只处理非空段落# 获取段落的最后一行lines = paragraph.text.split('\n')last_line = lines[-1] if lines else ""# 计算最后一行的字符宽度current_width = sum(2 if ord(c) > 127 else 1 for c in last_line)# 计算需要填充的全角空格数fill_count = max(0, target_length - current_width)# 计算需要填充的全角空格数fill_count = max(0, target_length - current_width)# print(f"段落内容: '{text}'")print(f"当前宽度: {current_width}, 需要填充空格数: {fill_count}\n")# 为原有内容添加实线下划线for run in paragraph.runs:run.font.underline = True  # 设置为单实线# 仅在最后添加全角空格并带下划线if fill_count > 0:# 清除原有换行符(如果有)if paragraph.runs and paragraph.runs[-1].text.endswith('\n'):paragraph.runs[-1].text = paragraph.runs[-1].text.rstrip('\n')fill_run = paragraph.add_run("\u3000"  * fill_count)  # 使用全角空格fill_run.font.underline = True# 恢复原有换行符(如果有)if len(lines) > 1:paragraph.add_run().add_break()# 保存文档doc.save(output_path)# 使用示例
path = r'C:\Users\jg2yXRZ\OneDrive\桌面\工具运用'
input_path = path + r'\附件1工具运用训会议记录(2025.4.23).docx'
output_path = path + r'\附件1工具运用训会议记录(2025.4.23)_实线下划线.docx'
add_solid_underline_with_spaces(input_path, output_path, target_length=29)
 
出现很多的方块
 

 
把全角改成空格
 

 

 
结果方块没有了,但是下划线不够长,
 

 
 

 
 分析:预设29个空格位置,已有字符超过29个,就不添加空格了
 

 
所以把预设空格数改到最大1000
 

 
运行后,都有空格了
 

 
结果显示
 

 
但是里面有大量的空格,会不会占空间?
 
结果显示:
 
如果设置1000字以上,空格太多,大小1.13
 
 

 
如果设置500字以上,大小变为22.3K
 

 
问题:
 
可是我并不知道每一段的总字数是多少,现在给所有段落默认添加1000字符,如果一段文字大于1000字,还是会没有空格和下划线。同时其他段落标题只有4-10个字,根本不需要填充1000字空格。
 
 
第2问:计算原有字符数+额外加多少空格下划线
 

 
 
 
 

 

 

 
 
手动测算一行有几个空
 

 
默认加78空,保证下划线肯定能撑满
 

 
 
代码展示
 
from docx import Documentz=78def add_dynamic_underline(doc_path, output_path, extra_length=z):"""为Word文档每段添加实线下划线,动态计算每段长度并额外增加指定字符数用全角空格填充至(实际长度+extra_length)的宽度参数:doc_path (str): 输入文档路径output_path (str): 输出文档路径extra_length (int): 每段额外增加的字符宽度"""# 加载文档doc = Document(doc_path)for paragraph in doc.paragraphs:if paragraph.text.strip():  # 只处理非空段落# 获取段落的最后一行lines = paragraph.text.split('\n')last_line = lines[-1] if lines else ""# 计算最后一行的字符宽度(中文算2,英文算1)current_width = sum(2 if ord(c) > 127 else 1 for c in last_line)# 动态设置目标长度 = 当前宽度 + 额外长度target_length = current_width + extra_length# 计算需要填充的全角空格数fill_count = max(0, target_length - current_width)print(f"段落内容: '{last_line}'")print(f"当前宽度: {current_width}, 目标宽度: {target_length}, 需要填充空格数: {fill_count}\n")# 为原有内容添加实线下划线for run in paragraph.runs:run.font.underline = True  # 设置为单实线# 仅在最后添加全角空格并带下划线if fill_count > 0:# 清除原有换行符(如果有)if paragraph.runs and paragraph.runs[-1].text.endswith('\n'):paragraph.runs[-1].text = paragraph.runs[-1].text.rstrip('\n')fill_run = paragraph.add_run(" " * fill_count)  # 使用全角空格fill_run.font.underline = True# 恢复原有换行符(如果有)if len(lines) > 1:paragraph.add_run().add_break()# 保存文档doc.save(output_path)# 使用示例
path = r'C:\Users\jg2yXRZ\OneDrive\桌面\工具运用'
input_path = path + r'\附件1工具运用训会议记录(2025.4.23).docx'
output_path = path + r'\附件1工具运用训会议记录(2025.4.23)_动态下划线.docx'
add_dynamic_underline(input_path, output_path, extra_length=z)
 
 
 
虽然打印时看不见,但我觉得右边距上一推灰点和回车,还是不好看,希望右边距上的点子不要出现。
 
 

 

 
第3问
 
计算一行可以有多少个字符
 

 
手动输入满一行
 

 
 
from docx import Documentdef count_width_per_line(doc_path):"""统计 Word 文档中每一行的宽度(中文=2,英文=1)"""doc = Document(doc_path)for i, paragraph in enumerate(doc.paragraphs, 1):if not paragraph.text.strip():  # 跳过空行continuelines = paragraph.text.split('\n')  # 按换行符分割for j, line in enumerate(lines, 1):if not line.strip():  # 跳过空行continue# 计算宽度(中文=2,英文=1)width = sum(2 if ord(c) > 127 else 1 for c in line)print(f"第 {i} 段,第 {j} 行 | 宽度: {width} | 字符数: {len(line)} | 内容: '{line}'")print("\n统计完成!")# 使用示例
path = r'C:\Users\jg2yXRZ\OneDrive\桌面\工具运用'
doc_path = path + r'\附件1工具运用训会议记录(2025.4.23).docx'
count_width_per_line(doc_path)
 
计算一行可以有多少个字符=69个
 

 
 
第4问:每段最后一行有有几个汉字,69减去最后一行的汉字的树数目就是添加额外的空格数量
 

 
 

 
 把都是汉字字符,是双数字符,尝试把69改成70
把都是汉字字符,是双数字符,尝试把69改成70
 
正好,都显示满了
 
 

 
代码展示
 
'''
会议记录自动补全下划线根据字数添加补全剩余空格下划线
deepseek \阿夏
20250424
'''
from docx import Document# 每行标准宽度(中文=2,英文=1)
h = 70def add_dynamic_underline(doc_path, output_path):"""为Word文档每段添加实线下划线,动态计算每段行数并调整下划线长度并保持原有字体格式参数:doc_path (str): 输入文档路径output_path (str): 输出文档路径"""doc = Document(doc_path)for paragraph in doc.paragraphs:if paragraph.text.strip():  # 只处理非空段落# 获取段落的最后一行lines = paragraph.text.split('\n')last_line = lines[-1] if lines else ""# 计算最后一行的字符宽度(中文算2,英文算1)current_width = sum(2 if ord(c) > 127 else 1 for c in last_line)# 计算该段有多少行(向上取整)line_count = current_width // hif line_count <1:line_count = 1extra_length = h - current_width else:extra_length = h - (current_width - line_count*h)print(f"段落内容: '{last_line}'")print(f"当前宽度: {current_width},汉字和英文的字符长度:'{len(last_line)}', 每行宽度: {h}, 行数: {line_count}")print(f"额外下划线长度: {extra_length}\n")# print(f"当前宽度: {current_width}, 目标宽度: {target_length}, 需要填充空格数: {fill_count}\n")# 为原有内容添加实线下划线for run in paragraph.runs:run.font.underline = True  # 设置为单实线# 仅在最后添加全角空格并带下划线if extra_length > 0:# 清除原有换行符(如果有)if paragraph.runs and paragraph.runs[-1].text.endswith('\n'):paragraph.runs[-1].text = paragraph.runs[-1].text.rstrip('\n')# 获取最后一个 run 的字体格式last_run = paragraph.runs[-1] if paragraph.runs else None# 添加空格,并复制原有格式fill_run = paragraph.add_run(" " * extra_length)if last_run:  # 如果存在原有 run,则复制其字体格式fill_run.font.name = last_run.font.name  # 字体fill_run.font.size = last_run.font.size  # 字号fill_run.font.bold = last_run.font.bold  # 加粗fill_run.font.italic = last_run.font.italic  # 斜体fill_run.font.underline = True  # 设置下划线# 恢复原有换行符(如果有)if len(lines) > 1:paragraph.add_run().add_break()# 保存文档doc.save(output_path)# 使用示例
path = r'C:\Users\jg2yXRZ\OneDrive\桌面\工具运用'
input_path = path + r'\附件1工具运用训会议记录(2025.4.23).docx'
output_path = path + r'\附件1工具运用训会议记录(2025.4.23)_动态下划线.docx'
add_dynamic_underline(input_path, output_path)
 
 
 
测试效果
 



 

 
删除空行,然后把内容变成10-1000字的段落
 

 

 

 
ok,完美实现补全下划线的目标。
 
 
后续做成exe,GUI界面,选择文件夹,发给不同的课题组成员补会议记录用(贴完文字后,用Python自动添加一下每行最后空白的下划线)让做的资料的文本版式更好看,