Python實(shí)現(xiàn)批量向PDF文件添加中文水印
前言
可以通過(guò)設(shè)置批量PDF文件所在的路徑及需要添加的水印名稱(chēng)可以實(shí)現(xiàn)批量添加PDF水印的效果。
實(shí)現(xiàn)思路是這樣的,通過(guò)在批量PDF文件路徑下面生成一個(gè)帶有水印的PDF模板。最后,將批量文件的每個(gè)PDF頁(yè)面和水印模板進(jìn)行合并完成批量添加水印的效果。

需要注意的是批量PDF文件必須和PDF模板水印文件的大小尺寸保持一致,這個(gè)可以在代碼里面調(diào)節(jié)一下就成了。
實(shí)現(xiàn)步驟
首先將需要添加水印的PDF文件準(zhǔn)備好放在一個(gè)文件夾下面。

在代碼中設(shè)置好PDF批量文件的路徑及水印名稱(chēng)。
if?__name__?==?'__main__':
????main('C:/pdf',?'我是一個(gè)水印')
內(nèi)部實(shí)現(xiàn)過(guò)程都封裝在main()函數(shù)里面了,這里改一下水印名稱(chēng)和批量PDF文件路徑直接執(zhí)行就好了。
啟動(dòng)以后,出現(xiàn)如下面的結(jié)果說(shuō)明已經(jīng)執(zhí)行完成了。

為了不覆蓋原來(lái)的PDF文件,合并后的文件都是添加了"已合并"字樣的PDF文件。

說(shuō)完了怎么操作,看一下主要的代碼塊部分有哪些吧。
其中用到的第三方庫(kù)有下面這些,里面我寫(xiě)了相關(guān)的注釋。
import?os??#?應(yīng)用文件操作
# reportlab是Python的一個(gè)標(biāo)準(zhǔn)庫(kù),可以畫(huà)圖、畫(huà)表格、編輯文字,最后可以輸出PDF格式。
from?reportlab.pdfgen?import?canvas
from?reportlab.lib.units?import?cm
from?reportlab.pdfbase?import?pdfmetrics
from?reportlab.pdfbase.ttfonts?import?TTFont
pdfmetrics.registerFont(TTFont('songti',?'C:/Windows/Fonts/simsun.ttc'))??#?加載宋體
# PyPDF2模塊主要的功能是分割或合并PDF文件,裁剪或轉(zhuǎn)換PDF文件中的頁(yè)面。
from?PyPDF2?import?PdfFileWriter,?PdfFileReader
import?logging??#?日志打印庫(kù)
日志模塊的初始化也比較簡(jiǎn)單,前面的文章中都有過(guò)相關(guān)的調(diào)用。
#?初始化日志設(shè)置
logger?=?logging.getLogger('批量添加水印')
logging.basicConfig(format='%(asctime)s?%(levelname)-8s:?%(message)s')
logger.setLevel(logging.DEBUG)
日志初始化完成后在后面需要打印日志的地方調(diào)用就可以了。
實(shí)現(xiàn)過(guò)程主要有三個(gè)函數(shù)來(lái)實(shí)現(xiàn)的,一個(gè)是為了生成水印模板、另一個(gè)是使水印模板和批量PDF文件執(zhí)行合并從而實(shí)現(xiàn)添加水印的功能、還有一個(gè)就是逐個(gè)遍歷批量PDF文件使其能夠逐個(gè)實(shí)現(xiàn)水印合并。
水印模板生成函數(shù)。
def?generate_water_pdf(content):
????'''
????生成帶有水印的PDF
????:param?content:?水印名稱(chēng)
????:return:
????'''
????cans?=?canvas.Canvas('water_back.pdf',?pagesize=(21?*?cm,?29.7?*?cm))
????cans.translate(10?*?cm,
???????????????????12?*?cm)??#?移動(dòng)原點(diǎn)坐標(biāo)
????cans.setFont('songti',?23)??#?設(shè)置字體為宋體、大小為23號(hào)
????cans.setFillColorRGB(0.5,?0.5,
?????????????????????????0.5)??#?設(shè)置字體背景顏色
????cans.rotate(45)??#?設(shè)置字體傾斜45度
????cans.drawString(-7?*?cm,?0?*?cm,?content)
????cans.drawString(7?*?cm,?0?*?cm,?content)
????cans.drawString(0?*?cm,?7?*?cm,?content)
????cans.drawString(0?*?cm,?-7?*?cm,?content)
????cans.save()??#?保存水印的PDF文件
水印合成實(shí)現(xiàn)函數(shù)。
def?insert_water_to_pdf(input_pdf,?output_pdf,?water_pdf): ????''' ????合并水印到PDF文件中 ????:param?input_pdf:?輸入文件路徑 ????:param?output_pdf:?輸出文件路徑 ????:param?water_pdf:?水印文件路徑 ????:return: ????''' ????water?=?PdfFileReader(water_pdf)??#?讀取水印PDF ????water_page?=?water.getPage(0)??#?獲取水印PDF的第一頁(yè) ????pdf?=?PdfFileReader(input_pdf,?strict=False)??#?讀取需要添加水印的文件 ????pdf_writer?=?PdfFileWriter()??#?創(chuàng)建PDF文件寫(xiě)入對(duì)象 ????for?page?in?range(pdf.getNumPages()):??#?遍歷每一頁(yè)P(yáng)DF對(duì)象 ????????pdf_page?=?pdf.getPage(page)??#?獲取PDF的當(dāng)前頁(yè)對(duì)象 ????????pdf_page.mergePage(water_page)??#?將水印頁(yè)合并到當(dāng)前頁(yè)中 ????????pdf_writer.addPage(pdf_page)??#?將合并后的PDF對(duì)象頁(yè)添加到PDF寫(xiě)入對(duì)象中 ????output_file?=?open(output_pdf,?'wb')??#?打開(kāi)PDF輸出文件 ????pdf_writer.write(output_file)??#?將文件寫(xiě)入到輸出文件 ????output_file.close()??#?關(guān)閉寫(xiě)入流
批量PDF文件遍歷調(diào)用合成函數(shù)。
def?main(diretory,?current):
????if?os.path.isdir(diretory):
????????logger.info('文件夾['?+?diretory?+?']校驗(yàn)成功!')
????????os.chdir(diretory)
????????logger.info('當(dāng)前路徑為['?+?os.getcwd()?+?']')
????????generate_water_pdf(current)
????????logger.info('水印PDF文件生成成功!')
????????for?file_path,?dir_names,?file_names?in?os.walk(r''?+?os.getcwd()):
????????????for?file_name?in?file_names:
????????????????try:
????????????????????name?=?file_name.split('.')[0]
????????????????????if?name?==?'water_back':
????????????????????????continue
????????????????????else:
????????????????????????file_name_path?=?os.path.join(file_path,?file_name)
????????????????????????output_file_path?=?file_name_path.split('.')[0]?+?'_已添加水印.pdf'
????????????????????????insert_water_to_pdf(file_name_path,?output_file_path,?'water_back.pdf')
????????????????????????logger.info('['?+?file_name_path?+?']完成水印合并!')
????????????????except?Exception?as?e:
????????????????????logger.error('['?+?file_name_path?+?']發(fā)生異常,執(zhí)行下一個(gè)!')
????????????????????logger.error('異常信息:'?+?repr(e))
????else:
????????logger.info('文件夾['?+?diretory?+?']校驗(yàn)失??!')
主要實(shí)現(xiàn)過(guò)程就是通過(guò)上面三個(gè)函數(shù)來(lái)完成的,最后調(diào)用后臺(tái)入口函數(shù)將mian()函數(shù)調(diào)用執(zhí)行就可以了。
完整代碼
# -*- coding:utf-8 -*-
# @author Python 集中營(yíng)
# @date 2022/1/27
# @file test4.py
# done
# 批量向PDF文件添加中文水印
import os # 應(yīng)用文件操作
# reportlab是Python的一個(gè)標(biāo)準(zhǔn)庫(kù),可以畫(huà)圖、畫(huà)表格、編輯文字,最后可以輸出PDF格式。
from reportlab.pdfgen import canvas
from reportlab.lib.units import cm
from reportlab.pdfbase import pdfmetrics
from reportlab.pdfbase.ttfonts import TTFont
pdfmetrics.registerFont(TTFont('songti', 'C:/Windows/Fonts/simsun.ttc')) # 加載宋體
# PyPDF2模塊主要的功能是分割或合并PDF文件,裁剪或轉(zhuǎn)換PDF文件中的頁(yè)面。
from PyPDF2 import PdfFileWriter, PdfFileReader
import logging # 日志打印庫(kù)
# 初始化日志設(shè)置
logger = logging.getLogger('批量添加水印')
logging.basicConfig(format='%(asctime)s %(levelname)-8s: %(message)s')
logger.setLevel(logging.DEBUG)
def generate_water_pdf(content):
'''
生成帶有水印的PDF
:param content: 水印名稱(chēng)
:return:
'''
cans = canvas.Canvas('water_back.pdf', pagesize=(21 * cm, 29.7 * cm))
cans.translate(10 * cm,
12 * cm) # 移動(dòng)原點(diǎn)坐標(biāo)
cans.setFont('songti', 23) # 設(shè)置字體為宋體、大小為23號(hào)
cans.setFillColorRGB(0.5, 0.5,
0.5) # 設(shè)置字體背景顏色
cans.rotate(45) # 設(shè)置字體傾斜45度
cans.drawString(-7 * cm, 0 * cm, content)
cans.drawString(7 * cm, 0 * cm, content)
cans.drawString(0 * cm, 7 * cm, content)
cans.drawString(0 * cm, -7 * cm, content)
cans.save() # 保存水印的PDF文件
def insert_water_to_pdf(input_pdf, output_pdf, water_pdf):
'''
合并水印到PDF文件中
:param input_pdf: 輸入文件路徑
:param output_pdf: 輸出文件路徑
:param water_pdf: 水印文件路徑
:return:
'''
water = PdfFileReader(water_pdf) # 讀取水印PDF
water_page = water.getPage(0) # 獲取水印PDF的第一頁(yè)
pdf = PdfFileReader(input_pdf, strict=False) # 讀取需要添加水印的文件
pdf_writer = PdfFileWriter() # 創(chuàng)建PDF文件寫(xiě)入對(duì)象
for page in range(pdf.getNumPages()): # 遍歷每一頁(yè)P(yáng)DF對(duì)象
pdf_page = pdf.getPage(page) # 獲取PDF的當(dāng)前頁(yè)對(duì)象
pdf_page.mergePage(water_page) # 將水印頁(yè)合并到當(dāng)前頁(yè)中
pdf_writer.addPage(pdf_page) # 將合并后的PDF對(duì)象頁(yè)添加到PDF寫(xiě)入對(duì)象中
output_file = open(output_pdf, 'wb') # 打開(kāi)PDF輸出文件
pdf_writer.write(output_file) # 將文件寫(xiě)入到輸出文件
output_file.close() # 關(guān)閉寫(xiě)入流
def main(diretory, current):
if os.path.isdir(diretory):
logger.info('文件夾[' + diretory + ']校驗(yàn)成功!')
os.chdir(diretory)
logger.info('當(dāng)前路徑為[' + os.getcwd() + ']')
generate_water_pdf(current)
logger.info('水印PDF文件生成成功!')
for file_path, dir_names, file_names in os.walk(r'' + os.getcwd()):
for file_name in file_names:
try:
name = file_name.split('.')[0]
if name == 'water_back':
continue
else:
file_name_path = os.path.join(file_path, file_name)
output_file_path = file_name_path.split('.')[0] + '_已添加水印.pdf'
insert_water_to_pdf(file_name_path, output_file_path, 'water_back.pdf')
logger.info('[' + file_name_path + ']完成水印合并!')
except Exception as e:
logger.error('[' + file_name_path + ']發(fā)生異常,執(zhí)行下一個(gè)!')
logger.error('異常信息:' + repr(e))
else:
logger.info('文件夾[' + diretory + ']校驗(yàn)失敗!')
if __name__ == '__main__':
main('C:/pdf', '我是一個(gè)水印')
到此這篇關(guān)于Python實(shí)現(xiàn)批量向PDF文件添加中文水印的文章就介紹到這了,更多相關(guān)Python PDF添加水印內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Python隨機(jī)生成身份證號(hào)碼及校驗(yàn)功能
這篇文章主要介紹了Python隨機(jī)生成身份證號(hào)碼及校驗(yàn)功能,文中給大家提到了校驗(yàn)碼計(jì)算方法,需要的朋友可以參考下2018-12-12
一起來(lái)學(xué)習(xí)一下python的數(shù)字類(lèi)型
這篇文章主要為大家詳細(xì)介紹了python的數(shù)字類(lèi)型,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-01-01
基于python實(shí)現(xiàn)語(yǔ)音錄入識(shí)別代碼實(shí)例
這篇文章主要介紹了如何通過(guò)python實(shí)現(xiàn)語(yǔ)音錄入識(shí)別,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01
基于Python編寫(xiě)一個(gè)網(wǎng)絡(luò)帶寬測(cè)試工具
這篇文章主要為大家詳細(xì)介紹了如何基于Python編寫(xiě)一個(gè)網(wǎng)絡(luò)帶寬測(cè)試工具,能夠準(zhǔn)確測(cè)量網(wǎng)絡(luò)的上傳和下載速度,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解下2025-12-12
Python中多進(jìn)程處理的Process和Pool的用法詳解
在Python編程中,多進(jìn)程是一種強(qiáng)大的并行處理技術(shù),Python提供了兩種主要的多進(jìn)程處理方式:Process和Pool,本文將詳細(xì)介紹這兩種方式的使用,希望對(duì)大家有所幫助2024-02-02
Python 實(shí)現(xiàn)一個(gè)顏色色值轉(zhuǎn)換的小工具
這篇文章主要介紹了Python 實(shí)現(xiàn)一個(gè)顏色色值轉(zhuǎn)換的小工具的相關(guān)資料,需要的朋友可以參考下2016-12-12
python實(shí)現(xiàn)公司年會(huì)抽獎(jiǎng)程序
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)公司年會(huì)抽獎(jiǎng)程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01

