Python 打包时包含字库文件的方法
在 Python 打包时,如果你想将字库文件(如 .ttf、.otf 等字体文件)一起打包成一个可执行文件,有几种常见的方法,具体取决于你使用的打包工具。
使用 PyInstaller
PyInstaller 是最常用的打包工具之一,以下是包含字库文件的方法:
1. 基本方法 - 使用 --add-data
参数:
pyinstaller --onefile --add-data="font.ttf;." your_script.py
多个:
pyinstaller --onefile --add-data="font1.ttf:." --add-data="font2.ttf:." --add-data="fonts/font3.otf:fonts/" your_script.py
-
使用 spec 文件(更推荐的方法):
-
首先生成 spec 文件:
pyi-makespec your_script.py
-
然后修改 spec 文件,在
a.datas
中添加字体文件:
-
a = Analysis(['your_script.py'],binaries=[],datas=[('font.ttf', '.')], # 添加这行...
)
添加多个文件
a = Analysis(['your_script.py'],binaries=[],datas=[('font1.ttf', '.'),('font2.ttf', '.'),('fonts/font3.otf', 'fonts'), # 保持目录结构('fonts/font4.ttf', 'fonts'),],...
)
需要在代码中处理字体路径:
import os
import sysdef resource_path(relative_path):""" 获取资源的绝对路径 """if hasattr(sys, '_MEIPASS'):return os.path.join(sys._MEIPASS, relative_path)return os.path.join(os.path.abspath("."), relative_path)# 使用示例
font_path = resource_path("font.ttf")
注意事项
-
确保在代码中正确引用字体文件路径(使用上述的
resource_path
方法) -
字体文件较大的话,打包后的文件体积会显著增加
-
测试打包后的程序是否能正确加载字体
-
考虑字体文件的许可证问题,确保你有权分发该字体
选择哪种方法取决于你的具体需求和使用的打包工具。PyInstaller 通常是跨平台打包的最佳选择。
批量添加字体文件夹
import glob# 获取所有字体文件
font_files = glob.glob('fonts/*.ttf') + glob.glob('fonts/*.otf')a = Analysis(['your_script.py'],binaries=[],datas=[(font, 'fonts') for font in font_files],...
)
注意事项
-
保持文件目录结构一致,特别是在代码中引用字体时
-
检查字体文件的许可证,确保可以合法分发
-
大量字体文件会显著增加打包后的体积
-
测试打包后的程序能否正确找到并加载所有字体
-
考虑使用相对路径而不是绝对路径引用字体文件
选择哪种方法取决于你的项目需求和使用的打包工具。对于大多数情况,PyInstaller 的 spec 文件方式提供了最好的灵活性和可维护性。
4. 使用 spec 文件打包
pyinstaller your_script.spec
5.在代码中访问打包后的字体文件
import os
import sys
from pathlib import Pathdef resource_path(relative_path):""" 获取打包后资源的绝对路径 """if hasattr(sys, '_MEIPASS'):base_path = Path(sys._MEIPASS)else:base_path = Path.cwd()return str(base_path / relative_path)# 使用示例
font_path = resource_path('fonts/NotoSans-Regular.ttf')# 使用字体(以Pillow为例)
from PIL import ImageFont
try:font = ImageFont.truetype(font_path, size=12)
except IOError:print(f"无法加载字体文件: {font_path}")font = ImageFont.load_default()