Python 打包时包含字库文件的方法

发布于:2025-05-12 ⋅ 阅读:(23) ⋅ 点赞:(0)

在 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
  1. 使用 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 sys

def 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")

注意事项

  1. 确保在代码中正确引用字体文件路径(使用上述的 resource_path 方法)

  2. 字体文件较大的话,打包后的文件体积会显著增加

  3. 测试打包后的程序是否能正确加载字体

  4. 考虑字体文件的许可证问题,确保你有权分发该字体

选择哪种方法取决于你的具体需求和使用的打包工具。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],
    ...
)

注意事项

  1. 保持文件目录结构一致,特别是在代码中引用字体时

  2. 检查字体文件的许可证,确保可以合法分发

  3. 大量字体文件会显著增加打包后的体积

  4. 测试打包后的程序能否正确找到并加载所有字体

  5. 考虑使用相对路径而不是绝对路径引用字体文件

选择哪种方法取决于你的项目需求和使用的打包工具。对于大多数情况,PyInstaller 的 spec 文件方式提供了最好的灵活性和可维护性。

4. 使用 spec 文件打包

pyinstaller your_script.spec

5.在代码中访问打包后的字体文件

import os
import sys
from pathlib import Path

def 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()


网站公告

今日签到

点亮在社区的每一天
去签到