1. 前言

上一篇文章,对 Word 写入数据的一些常见操作进行了总结

最全总结 | 聊聊 Python 办公自动化之 Word(上)

相比写入数据,读取数据同样很实用!

本篇文章,将谈谈如何全面读取一个 Word 文档中的数据,并会指出一些要注意的点

2. 基本信息

我们同样使用 python-docx 这个依赖库来对 Word 文档进行读取

首先我们来读取文档的基本信息

它们分别是:章节、页边距、页眉页脚边距、页面宽高、页面方向等

在获取文档基础信息之前,我们通过文档路径构建一个文档对象 Document

  1. from docx import Document
  2. # 源文件目录
  3. self.word_path = './output.docx'
  4. # 打开文档,构建一个文档对象
  5. self.doc = Document(self.word_path)

1 - 章节( Section )

  1. # 1、获取章节信息
  2. # 注意:章节可以设置本页的大小、页眉、页脚
  3. msg_sections = self.doc.sections
  4. print("章节列表:", msg_sections)
  5. # 章节数目
  6. print('章节数目:', len(msg_sections))

2 - 页边距( Page Margin )

通过章节对象的 left_margin、top_margin、right_margin、bottom_margin 属性值可以获取当前章节的左边距、上边距、右边距、下边距

  1. def get_page_margin(section):
  2. """
  3. 获取某个页面的页边距(EMU)
  4. :param section:
  5. :return:
  6. """
  7. # 分别对应:左边距、上边距、右边距、下边距
  8. left, top, right, bottom = section.left_margin, section.top_margin, section.right_margin, section.bottom_margin
  9. return left, top, right, bottom
  10. # 2、页边距信息
  11. first_section = msg_sections[0]
  12. left, top, right, bottom = get_page_margin(first_section)
  13. print('左边距:', left, ",上边距:", top, ",右边距:", right, ",下边距:", bottom)

返回值的单位是 EMU,和厘米、英尺的转换关系如下:

3 - 页眉页脚边距

页眉边距:header_distance

页脚边距:footer_distance

  1. def get_header_footer_distance(section):
  2. """
  3. 获取页眉、页脚边距
  4. :param section:
  5. :return:
  6. """
  7. # 分别对应页眉边距、页脚边距
  8. header_distance, footer_distance = section.header_distance, section.footer_distance
  9. return header_distance, footer_distance
  10. # 3、页眉页脚边距
  11. header_distance, footer_distance = get_header_footer_distance(first_section)
  12. print('页眉边距:', header_distance, ",页脚边距:", footer_distance)

4 - 页面宽度和高度

页面宽度:page_width

页面高度:page_height

  1. def get_page_size(section):
  2. """
  3. 获取页面宽度、高度
  4. :param section:
  5. :return:
  6. """
  7. # 分别对应页面宽度、高度
  8. page_width, page_height = section.page_width, section.page_height
  9. return page_width, page_height
  10. # 4、页面宽度、高度
  11. page_width, page_height = get_page_size(first_section)
  12. print('页面宽度:', page_width, ",页面高度:", page_height)

5 - 页面方向( Page Orientation )

页面方向分为:横向和纵向

使用章节对象的 orientation 属性去获取一个章节的页面方向

  1. def get_page_orientation(section):
  2. """
  3. 获取页面方向
  4. :param section:
  5. :return:
  6. """
  7. return section.orientation
  8. # 5、页面方向
  9. # 类型:class 'docx.enum.base.EnumValue
  10. # 包含:PORTRAIT (0)、LANDSCAPE (1)
  11. page_orientation = get_page_orientation(first_section)
  12. print("页面方向:", page_orientation)

同样,可以直接使用这个属性设置一个章节的方向

  1. from docx.enum.section import WD_ORIENT
  2. # 设置页面方向(横向、竖向)
  3. # 设置为横向
  4. first_section.orientation = WD_ORIENT.LANDSCAPE
  5. # 设置为竖向
  6. # first_section.orientation = WD_ORIENT.PORTRAIT
  7. self.doc.save(self.word_path)

3. 段落

使用文档对象的 paragraphs 属性可以获取文档中所有的段落

注意:这里获取的段落不包含页眉、页脚、表格中的段落

  1. # 获取文档对象中所有的段落,默认不包含:页眉、页脚、表格中的段落
  2. paragraphs = self.doc.paragraphs
  3. # 1、段落数目
  4. paragraphs_length = len(paragraphs)
  5. print('文档中一共包含:{}个段落'.format(paragraphs_length))

1 - 段落内容

我们可以遍历文档中所有的段落列表,通过段落对象的 text 属性,获取全部的段落内容

  1. # 0、读取所有段落数据
  2. contents = [paragraph.text for paragraph in self.doc.paragraphs]
  3. print(contents)

2 - 段落格式

通过上一篇文章,我们知道段落也存在格式的

使用 paragraph_format 属性获取段落的基本格式信息

包含:对齐方式、左右缩进、行间距、段落前后间距等

  1. # 2、获取某一个段落的格式信息
  2. paragraph_someone = paragraphs[0]
  3. # 2.1 段落内容
  4. content = paragraph_someone.text
  5. print('段落内容:', content)
  6. # 2.2 段落格式
  7. paragraph_format = paragraph_someone.paragraph_format
  8. # 2.2.1 对齐方式
  9. # <class 'docx.enum.base.EnumValue'>
  10. alignment = paragraph_format.alignment
  11. print('段落对齐方式:', alignment)
  12. # 2.2.2 左、右缩进
  13. left_indent, right_indent = paragraph_format.left_indent, paragraph_format.right_indent
  14. print('段落左缩进:', left_indent, ",右缩进:", right_indent)
  15. # 2.2.3 首行缩进
  16. first_line_indent = paragraph_format.first_line_indent
  17. print('段落首行缩进:', first_line_indent)
  18. # 2.2.4 行间距
  19. line_spacing = paragraph_format.line_spacing
  20. print('段落行间距:', line_spacing)
  21. # 2.2.5 段落前后间距
  22. space_before, space_after = paragraph_format.space_before, paragraph_format.space_after
  23. print('段落前、后间距分别为:', space_before, ',', space_after)

4. 文字块 - Run

文字块 Run 属于段落的一部分,所以,要获取文字块信息,必须先拿到一个段落实例对象

以文字块基本信息、字体格式信息为例

1 - 文字块基本信息

我们使用段落对象的 runs 属性获取段落内所有的文字块对象

  1. def get_runs(paragraph):
  2. """
  3. 获取段落下所有的文字块信息,包含:数目、内容列表
  4. :param paragraph:
  5. :return:
  6. """
  7. # 段落对象包含的文字块Run
  8. runs = paragraph.runs
  9. # 数量
  10. runs_length = len(runs)
  11. # 文字块内容
  12. runs_contents = [run.text for run in runs]
  13. return runs, runs_length, runs_contents

2 - 文字块格式信息

文字块是文档中最小的文字单元,使用文字块对象的 font 属性可以拿到它的字体属性

和设置文字块格式属性一一对应,字体名称、大小、颜色、是否加粗、是否斜体等都可以获取到

  1. # 2、文字块格式信息
  2. # 包含:字体名称、大小、颜色、是否加粗等
  3. # 某一个文字块的字体属性
  4. run_someone_font = runs[0].font
  5. # 字体名称
  6. font_name = run_someone_font.name
  7. print('字体名称:', font_name)
  8. # 字体颜色(RGB)
  9. # <class 'docx.shared.RGBColor'>
  10. font_color = run_someone_font.color.rgb
  11. print('字体颜色:', font_color)
  12. print(type(font_color))
  13. # 字体大小
  14. font_size = run_someone_font.size
  15. print('字体大小:', font_size)
  16. # 是否加粗
  17. # True:加粗;None/False:没有加粗
  18. font_bold = run_someone_font.bold
  19. print('是否加粗:', font_bold)
  20. # 是否斜体
  21. # True:协议;None/False:不是斜体
  22. font_italic = run_someone_font.italic
  23. print('是否斜体:', font_italic)
  24. # 带下划线
  25. # True:带有下滑线;None/False:字体没有带下滑线
  26. font_underline = run_someone_font.underline
  27. print('带有下滑线:', font_underline)
  28. # 删除线/双删除线
  29. # True:带有删除线;None/False:字体没有带删除线
  30. font_strike = run_someone_font.strike
  31. font_double_strike = run_someone_font.double_strike
  32. print('带有删除线:', font_strike, "\n带有双删除线:", font_double_strike)

5. 表格

文档对象的 tables 属性可以获取当前文档中所有的表格对象

  1. # 文档中所有的表格对象
  2. tables = self.doc.tables
  3. # 1、表格数量
  4. table_num = len(tables)
  5. print('文档中包含的表格数量:', table_num)

1 - 表格所有数据

获取表格中所有数据有 2 种方式

第一种方式:通过遍历文档中所有表格,然后按行和单元格进行遍历,最后通过单元格的 text 属性获取所有单元格的文本内容

  1. # 2、读取所有表格数据
  2. # 所有表格对象
  3. # tables = [table for table in self.doc.tables]
  4. print('内容分别是:')
  5. for table in tables:
  6. for row in table.rows:
  7. for cell in row.cells:
  8. print(cell.text, end=' ')
  9. print()
  10. print('\n')

另外一种方式是使用表格对象的 _cells 属性获取表格中所有的单元格,然后遍历获取单元格的值

  1. def get_table_cell_content(table):
  2. """
  3. 读取表格中所有单元格是内容
  4. :param table:
  5. :return:
  6. """
  7. # 所有单元格
  8. cells = table._cells
  9. cell_size = len(cells)
  10. # 所有单元格的内容
  11. content = [cell.text for cell in cells]
  12. return content

2 - 表格样式

  1. # 3、表格样式名称
  2. # Table Grid
  3. table_someone = tables[0]
  4. style = table_someone.style.name
  5. print("表格样式:", style)

3 - 表格行数量、列数量

table.rows:表格中的行数据迭代对象

table.columns:表格中的列数据迭代对象

  1. def get_table_size(table):
  2. """
  3. 获取表格的行数量、列数量
  4. :param table:
  5. :return:
  6. """
  7. # 几行、几列
  8. row_length, column_length = len(table.rows), len(table.columns)
  9. return row_length, column_length

4 - 行数据、列数据

有时候,我们需要单独按照行或者列,获取全部数据

  1. def get_table_row_datas(table):
  2. """
  3. 获取表格中行数据
  4. :param table:
  5. :return:
  6. """
  7. rows = table.rows
  8. datas = []
  9. # 每一行获取单元格的数据组成列表,加入到结果列表中
  10. for row in rows:
  11. datas.append([cell.text for cell in row.cells])
  12. return datas
  13. def get_table_column_datas(table):
  14. """
  15. 获取表格中列数据
  16. :param table:
  17. :return:
  18. """
  19. columns = table.columns
  20. datas = []
  21. # 每一列获取单元格的数据组成列表,加入到结果列表中
  22. for column in columns:
  23. datas.append([cell.text for cell in column.cells])
  24. return datas

6. 图片

有时候,我们需要将 Word 文档中的图片下载到本地

Word 文档实际上也是一个压缩文件,我们使用解压工具后发现,文档包含的图片都放置在 /word/media/ 目录下

提取文档图片有 2 种方法,分别是:

  • 解压文档文件,将对应目录下的图片拷贝出来

  • 使用 python-docx 内置的方法提取图片( 推荐 )

  1. def get_word_pics(doc, word_path, output_path):
  2. """
  3. 提取word文档内的图片
  4. :param word_path:源文件名称
  5. :param output_path: 结果目录
  6. :return:
  7. """
  8. dict_rel = doc.part._rels
  9. for rel in dict_rel:
  10. rel = dict_rel[rel]
  11. if "image" in rel.target_ref:
  12. # 图片保存目录
  13. if not os.path.exists(output_path):
  14. os.makedirs(output_path)
  15. img_name = re.findall("/(.*)", rel.target_ref)[0]
  16. word_name = os.path.splitext(word_path)[0]
  17. # 新的名称
  18. newname = word_name.split('\\')[-1] if os.sep in word_name else word_name.split('/')[-1]
  19. img_name = f'{newname}_{img_name}'
  20. # 写入到文件中
  21. with open(f'{output_path}/{img_name}', "wb") as f:
  22. f.write(rel.target_part.blob)

7. 页眉页脚

页眉和页脚都是基于章节

我们以某一个章节对象为例进行说明

  1. ​# 获取某一个章节
  2. first_section = self.doc.sections[0]

使用章节对象的 header、footer 属性可以获取页眉、页脚对象

由于页眉、页脚可能包含多个段落 Paragraph,因此,我们可以先使用页眉页脚对象的 paragraphs 属性获取所有段落,然后遍历出所有段落的值,最后拼接起来就是页眉页脚的全部内容

  1. # 注意:页眉、页脚都有可能包含多个段落
  2. # 页眉所有的段落
  3. header_content = " ".join([paragraph.text for paragraph in first_section.header.paragraphs])
  4. print("页眉内容:", header_content)
  5. # 页脚
  6. footer_content = " ".join([paragraph.text for paragraph in first_section.footer.paragraphs])
  7. print("页脚内容:", footer_content)

8. 最后

本篇文章和上一篇文章,分别对 Word 文档的读写进行了一次全面讲解

文中全部源码我已经上传到后台,关注公众号「 AirPython 」,回复「 word 」即可获得全部源码

如果你觉得文章还不错,请大家 点赞、分享、留言下,因为这将是我持续输出更多优质文章的最强动力!

推荐阅读

最全总结 | 聊聊 Python 办公自动化之 Excel(上)

最全总结 | 聊聊 Python 办公自动化之 Excel(中)

最全总结 | 聊聊 Python 办公自动化之 Excel(下)

最全总结 | 聊聊 Python 办公自动化之 Word(上)

最全总结 | 聊聊 Python 办公自动化之 Word(中)的更多相关文章

  1. 最全总结 | 聊聊 Python 办公自动化之 Word(下)

    1. 前言 关于 Word 文档的读写,前面两篇文章分别进行了一次全面的总结 最全总结 | 聊聊 Python 办公自动化之 Word(上) 最全总结 | 聊聊 Python 办公自动化之 Word( ...

  2. 最全总结 | 聊聊 Python 办公自动化之 Word(上)

    1. 前言 日常自动化办公中,使用 Python 真的能做到事半功倍! 在上一个系列中,我们对 Python 操作 Excel 进行了一次全面总结 最全总结 | 聊聊 Python 办公自动化之 Ex ...

  3. 最全总结 | 聊聊 Python 办公自动化之 PDF(上)

    1. 前言 自动化办公,非 Python 莫属! 从本篇文章开始,我们继续聊聊自动化办公中另外一个常用系列:PPT 2. 准备一下 Python 操作 PPT 最强大的依赖库是:python-pptx ...

  4. 最全总结 | 聊聊 Python 办公自动化之 PPT(中)

    1. 前言 上一篇文章简单地介绍了 PPT 的文档结构,并使用 python-pptx 这个依赖库完成对 PPT 文档最基本的操作 最全总结 | 聊聊 Python 办公自动化之 PPT(上) 作为 ...

  5. 最全总结 | 聊聊 Python 办公自动化之 Excel(中)

    1. 前言 上一篇文章中,我们聊到使用 xlrd.xlwt.xlutils 这一组合操作 Excel 的方法 最全总结 | 聊聊 Python 办公自动化之 Excel(上) ​本篇文章将继续聊另外一 ...

  6. 最全总结 | 聊聊 Python 办公自动化之 Excel(下)

    1. 前言 前面谈到 Python 处理 Excel 文件最常见的两种方式,即:xlrd/xlwt.openpyxl ​其中, xlrd/xlwt 这一组合,xlrd 可以负责读取数据,而 xlwt ...

  7. 最全总结 | 聊聊 Python 办公自动化之 Excel(上)

    1. 前言 在我们日常工作中,经常会使用 Word.Excel.PPT.PDF 等办公软件 但是,经常会遇到一些重复繁琐的事情,这时候手工操作显得效率极其低下:通过 Python 实现办公自动化变的很 ...

  8. 最全总结 | 聊聊 Python 数据处理全家桶(PgSQL篇)

    1. 前言 大家好,我是安果! Python 数据处理全家桶,截止到现在,一共写过 6 篇文章,有兴趣的小伙伴可以去了解一下! 最全总结 | 聊聊 Python 数据处理全家桶(Mysql 篇) 最全 ...

  9. 最全总结 | 聊聊 Python 数据处理全家桶(Mysql 篇)

    1. 前言 在爬虫.自动化.数据分析.软件测试.Web 等日常操作中,除 JSON.YAML.XML 外,还有一些数据经常会用到,比如:Mysql.Sqlite.Redis.MongoDB.Memch ...

随机推荐

  1. 如何写好 C语言 main 函数!你准备好编写 C 程序了吗?

    学习如何构造一个 C 文件并编写一个 C main 函数来成功地处理命令行参数.   我知道,现在孩子们用 Python 和 JavaScript 编写他们的疯狂"应用程序".但是 ...

  2. docker启动服务---------------mysql

    1.查找镜像: docker search mysql 也可以去官网查看镜像tag,选择自己需要的版本,否则会下载最新版本:https://hub.docker.com/_/mysql/ 2.下载镜像 ...

  3. linux(centos8):安装java jdk 15 (java 15)

    一,下载jdk15 官方网站: https://www.oracle.com/java/ 下载页面: https://www.oracle.com/cn/java/technologies/javas ...

  4. linux(centos8):使用zip/unzip压缩和解压缩文件

    一,查看zip命令所属的rpm包 1,zip [root@kubemaster ~]# whereis zip zip: /usr/bin/zip /usr/share/man/man1/zip.1. ...

  5. 解决加密PDF文档无法复制文字的问题

    有的时候在网络上搜索到一篇心仪的PDF文档,想复制其中内容时提示无法复制. 如果只想摘抄其中部分文字内容,可以使用Firefox浏览器打开这篇加密文档. Firefox浏览器自带PDF插件,打开后即可 ...

  6. 源码安装中./configure的使用

    在linux中安装源码,在源码目录下使用 ./configure --prefix=xxx  --with=xxx 其中configure是一个可执行脚本, --prefix 选项就是配置安装的路径, ...

  7. javascript模块化(简)

    这里书写一个个人理解以及整理的东西,关于模块化以及ES6语法推荐大家阅读阮一峰老师的ES6入门教程 地址:https://es6.ruanyifeng.com/ 比较散,请见谅 以前的js是没有模块化 ...

  8. java安全编码指南之:文件IO操作

    目录 简介 创建文件的时候指定合适的权限 注意检查文件操作的返回值 删除使用过后的临时文件 释放不再被使用的资源 注意Buffer的安全性 注意 Process 的标准输入输出 InputStream ...

  9. 从Linux源码看TIME_WAIT状态的持续时间

    从Linux源码看TIME_WAIT状态的持续时间 前言 笔者一直以为在Linux下TIME_WAIT状态的Socket持续状态是60s左右.线上实际却存在TIME_WAIT超过100s的Socket ...

  10. 配置交换机之间直连链路聚合-LACP模式

    组网图形 LACP模式链路聚合简介 以太网链路聚合是指将多条以太网物理链路捆绑在一起成为一条逻辑链路,从而实现增加链路带宽的目的.链路聚合分为手工模式和LACP模式. LACP模式需要有链路聚合控制协 ...