介绍

官网

PyInstaller将 Python 应用程序及其所有依赖项捆绑到一个包中。用户无需安装 Python 解释器或任何模块即可运行打包的应用程序。 PyInstaller支持 Python 3.6 或更新版本

PyInstaller 完全兼容 Windows、Mac OS X 和 GNU/Linux 。并且使用都是一致的。

安装

pyinstaller 在各平台安装方式是一样的

pip install pyinstaller

使用

使用 pyinstaller 打包后 会在项目目录下创建一个 build 目录和一个 dist 目录

打包好的可执行文件(Windwos: .exe , linux: )在 dist 目录中

命令行参数

pyinstaller [选项] [脚本]  | 规范文件# 打包时使用的参数
-H 帮助信息
-v 版本信息
--distpath 打包好的可执行程序存放目录,默认是 ./dist
--workpath 打包时临时文件存放目录,默认是 ./build
--clean  在构建打包之前清理缓存兵删除临时文件
--log-level [level]  构建时控制台输出日志信息的等级。默认为INFO,可选 TRACE、DEBUG、INFO、WARN、ERROR、CRITICAL# 生成可执行文件的参数
-F [filename.py]  创建一个单文件的可执行程序
-D [filename.py] 创建一个文件夹,文件夹内包含可执行程序和其他依赖文件。这种方式不能把可执行成功移动到其他路径
--specpath [规范文件目录]  指定规范文件存放位置
-n [name] 指定可执行文件名称。默认是第一个脚本的名称
-d  debug模式# 捆绑选项,添加其他文件
--add-data ["源地址;目标地址"] 要添加到可执行文件内的其他非二进制文件或文件夹。此选项可以多次使用
windows 使用: --add-data="txt;."
linux 使用:--add-data="*.txt:."
打包后需要把此选项添加的文件复制一份到目标地址(可执行文件同目录下)--add-binary [binaru-file] 要添加到可执行文件内的二进制文件。此选项可多次使用
-p [path] 搜索导入的路径。以':'分隔,或多次使用此选项--hidden-import MODULENAME , --hiddenimport MODULENAME
命名在脚本代码中不可见的导入。此选项可以多次使用。--collect-submodules MODULENAME
从指定的包或模块中收集所有子模块。此选项可以多次使用。--collect-data MODULENAME , --collect- datas MODULENAME
从指定的包或模块收集所有数据。此选项可以多次使用。--collect-binaries MODULENAME
从指定的包或模块收集所有二进制文件。此选项可以多次使用。--collect-all MODULENAME
从指定的包或模块中收集所有子模块、数据文件和二进制文件。此选项可以多次使用。--copy-metadata包名
复制指定包的元数据。此选项可以多次使用。--recursive-copy-metadata包名
复制指定包及其所有依赖项的元数据。此选项可以多次使用。--additional-hooks-dir HOOKSPATH
搜索钩子的附加路径。此选项可以多次使用。--runtime钩RUNTIME_HOOKS
自定义运行时挂钩文件的路径。运行时挂钩是与可执行文件捆绑在一起的代码,它在任何其他代码或模块之前执行,以设置运行时环境的特殊功能。此选项可以多次使用。--exclude-module排除
将被忽略的可选模块或包(Python 名称,而不是路径名称)(好像没有找到)。此选项可以多次使用。--key键
用于加密 Python 字节码的密钥。--splash IMAGE_FILE
(实验)向应用程序添加带有图像 IMAGE_FILE 的启动画面。启动画面可以在解包时显示进度更新。# Windows 和 Mac 特定选项
-c 控制台选项
-w 窗口化选项 需要有ui页面
-i [icon path] 指定可执行文件的icon

Tips:

1、打包文件有多个依赖文件的时候,导包的时候只能从项目目录下开始。例如

2、打包多个文件时,需要把所有py文件都写上;有其他类型的依赖文件比如readme,配置文件,数据文件等要使用 --add-data [file] 添加

pyinstaller -F demo_a_ab.py demo_a_aa.py get_demos/get_one.py --clean

更多

如果以上介绍没有你要找的内容,请看以下转载自 Pyinstaller 打包发布经验总结 的内容

使用Pyinstaller打包Python项目包含了大量的坑,这篇文章总结实践得到的Pyinstaller打包经验。本文的例子为Python3.6代码,Pyinstaller3.4,在windows下打包为64位和32位版本。

目录

Pyinstaller基本使用方法

Python项目的打包方法

1.spec文件生成

2.spec文件配置

3.使用spec执行打包命令

Visual C++ run-time .dlls包含

Python模块的打包问题

冻结打包路径

其它问题

Pyinstaller基本使用方法

Python项目的打包方法

1.spec文件生成

2.spec文件配置

3.使用spec执行打包命令

Python模块的打包问题

冻结打包路径

其它问题

Pyinstaller基本使用方法
Pyinstaller可以通过简单的命令进行python代码的打包工作,其基本的命令为:

pyinstaller -option xxx.py
options的详情可参考官方帮助文档https://pyinstaller.readthedocs.io/en/stable/usage.html

这边只介绍用到的option:-d生成一个文件目录包含可执行文件和相关动态链接库和资源文件等;-f仅生成一个可执行文件

-D, --onedir    Create a one-folder bundle containing an executable (default)
-F, --onefile    Create a one-file bundled executable.
对于打包结果较大的项目,选用-d生成目录相比单可执行文件的打包方式,执行速度更快,但包含更加多的文件。本文的例子选中-D方式打包。

Python项目的打包方法
以一个多文件和目录的Python项目为例,项目文件包含:1.Python源代码文件;2.图标资源文件;3.其它资源文件

以图中项目为例,Python源代码文件在多个目录下:bin, lib\app, lib\models, lib\views;图标资源文件在lib\icon目录下;其它资源文件在data目录下,包括文本文件,视频文件等等。

1.spec文件生成
为了进行自定义配置的打包,首先需要编写打包的配置文件.spec文件。当使用pyinstaller -d xxx.py时候会生成默认的xxx.spec文件进行默认的打包配置。通过配置spec脚本,并执行pyinstaller -d xxx.spec完成自定义的打包。

通过生成spec文件的命令,针对代码的主程序文件生成打包对应的spec文件

pyi-makespec -w xxx.py
打开生成的spec文件,修改其默认脚本,完成自定义打包需要的配置。spec文件是一个python脚本,其默认的结构如下例所示

# -*- mode: python -*-block_cipher = Nonea = Analysis(['fastplot.py'],pathex=['D:\\install_test\\DAGUI-0.1\\bin'],binaries=[],datas=[],hiddenimports=[],hookspath=[],runtime_hooks=[],excludes=[],win_no_prefer_redirects=False,win_private_assemblies=False,cipher=block_cipher)
pyz = PYZ(a.pure, a.zipped_data,cipher=block_cipher)
exe = EXE(pyz,a.scripts,exclude_binaries=True,name='fastplot',debug=False,strip=False,upx=True,console=False )
coll = COLLECT(exe,a.binaries,a.zipfiles,a.datas,strip=False,upx=True,name='fastplot')

spec文件中主要包含4个class: Analysis, PYZ, EXE和COLLECT.

Analysis以py文件为输入,它会分析py文件的依赖模块,并生成相应的信息

PYZ是一个.pyz的压缩包,包含程序运行需要的所有依赖

EXE根据上面两项生成

COLLECT生成其他部分的输出文件夹,COLLECT也可以没有

2.spec文件配置
首先给出举例python项目的spec文件配置

# -*- mode: python -*-
import sys
import os.path as osp
sys.setrecursionlimit(5000)block_cipher = NoneSETUP_DIR = 'D:\\install_test\\FASTPLOT\\'a = Analysis(['fastplot.py','frozen_dir.py','D:\\install_test\\FASTPLOT\\lib\\app\\start.py','D:\\install_test\\FASTPLOT\\lib\\models\\analysis_model.py','D:\\install_test\\FASTPLOT\\lib\\models\\datafile_model.py','D:\\install_test\\FASTPLOT\\lib\\models\\data_model.py','D:\\install_test\\FASTPLOT\\lib\\models\\figure_model.py','D:\\install_test\\FASTPLOT\\lib\\models\\time_model.py','D:\\install_test\\FASTPLOT\\lib\\models\\mathematics_model.py','D:\\install_test\\FASTPLOT\\lib\\views\\constant.py','D:\\install_test\\FASTPLOT\\lib\\views\\custom_dialog.py','D:\\install_test\\FASTPLOT\\lib\\views\\data_dict_window.py','D:\\install_test\\FASTPLOT\\lib\\views\\data_process_window.py','D:\\install_test\\FASTPLOT\\lib\\views\\data_sift_window.py','D:\\install_test\\FASTPLOT\\lib\\views\\mathematics_window.py','D:\\install_test\\FASTPLOT\\lib\\views\\para_temp_window.py','D:\\install_test\\FASTPLOT\\lib\\views\\mainwindow.py','D:\\install_test\\FASTPLOT\\lib\\views\\paralist_window.py','D:\\install_test\\FASTPLOT\\lib\\views\\plot_window.py'],pathex=['D:\\install_test\\FASTPLOT'],binaries=[],datas=[(SETUP_DIR+'lib\\icon','lib\\icon'),(SETUP_DIR+'data','data')],hiddenimports=['pandas','pandas._libs','pandas._libs.tslibs.np_datetime','pandas._libs.tslibs.timedeltas','pandas._libs.tslibs.nattype', 'pandas._libs.skiplist','scipy._lib','scipy._lib.messagestream'],hookspath=[],runtime_hooks=[],excludes=[],win_no_prefer_redirects=False,win_private_assemblies=False,cipher=block_cipher)pyz = PYZ(a.pure, a.zipped_data,cipher=block_cipher)
exe = EXE(pyz,a.scripts,exclude_binaries=True,name='fastplot',debug=False,strip=False,upx=True,console=True)
coll = COLLECT(exe,a.binaries,a.zipfiles,a.datas,strip=False,upx=True,name='fastplot')

a) py文件打包配置

针对多目录多文件的python项目,打包时候需要将所有相关的py文件输入到Analysis类里。Analysis类中的pathex定义了打包的主目录,对于在此目录下的py文件可以只写文件名不写路径。如上的spec脚本,将所有项目中的py文件路径以列表形式写入Analysis,这里为了说明混合使用了绝对路径和相对路径。

b) 资源文件打包配置

资源文件包括打包的python项目使用的相关文件,如图标文件,文本文件等。对于此类资源文件的打包需要设置Analysis的datas,如例子所示datas接收元组:datas=[(SETUP_DIR+'lib\\icon','lib\\icon'),(SETUP_DIR+'data','data')]。元组的组成为(原项目中资源文件路径,打包后路径),例子中的(SETUP_DIR+'lib\\icon','lib\\icon')表示从D:\\install_test\\FASTPLOT\\lib\\icon下的图标文件打包后放入打包结果路径下的lib\\icon目录。

c)Hidden import配置

pyinstaller在进行打包时,会解析打包的python文件,自动寻找py源文件的依赖模块。但是pyinstaller解析模块时可能会遗漏某些模块(not visible to the analysis phase),造成打包后执行程序时出现类似No Module named xxx。这时我们就需要在Analysis下hiddenimports中加入遗漏的模块,如例子中所示。

d)递归深度设置

在打包导入某些模块时,常会出现"RecursionError: maximum recursion depth exceeded"的错误,这可能是打包时出现了大量的递归超出了python预设的递归深度。因此需要在spec文件上添加递归深度的设置,设置一个足够大的值来保证打包的进行,即

import sys
sys.setrecursionlimit(5000)

e)去除不必要的模块import

有时需要让pyinstaller不打包某些用不到的模块,可通过在excludes=[]中添加此模块实现,如

excludes=['zmq']

3.使用spec执行打包命令

pyinstaller -D xxx.spec

打包生成两个文件目录build和dist,build为临时文件目录完成打包后可以删除;dist中存放打包的结果,可执行文件和其它程序运行的关联文件都在这个目录下。

Visual C++ run-time .dlls包含
针对在Windows<10发布使用,且Python>=3.5的情况,Pyinstaller打包的程序可能会出现不包含Visual C++ run-time .dlls的情况,Python>=3.5需要使用Visual Studio 2015 run-time,也就是Universal CRT,这些runtime在Win10本身或Win7到Win8.1版本的更新包里,但程序打包后使用的系统里并不一定安装了。因此需要参考Universal CRT的建议,应用以下的方法解决这个问题:

Build on Windows 7 which has been reported to work.

Include one of the VCRedist packages (the redistributable package files) into your application’s installer. This is Microsoft’s recommended way, see “Distributing Software that uses the Universal CRT“ in the above-mentioned link, numbers 2 and 3.

Install the Windows Software Development Kit (SDK) for Windows 10 and expand the .spec-file to include the required DLLs, see “Distributing Software that uses the Universal CRT“ in the above-mentioned link, number 6.

Python模块的打包问题
程序调用的很多包,在打包时候可能会出现一些问题,针对这写问题需要做一些处理才能保证打包的程序正常执行。

1.PyQt plugins缺失

使用PyQt编写UI交互界面的python代码在进行打包时可能会出现一些特别的问题。

执行使用了PyQt的打包程序,常会出现这样的错误,提示缺少Qt platfrom plugin “windows”,如下图

打包后程序运行后,使用png格式的图标可以正常显示,但使用的ico格式图标不显示(对于所有图标和关联文件都无法使用的情况涉及到路径问题,后文会另外解释)。

这两个错误产生的问题都是因为打包时没有将PyQt相关的动态链接库目录生成到打包目录下,因此可以通过将这些需要的文件目录拷贝到打包生成目录下,解决plugin缺失问题。以使用PyQt5编写的python软件打包为例,完成打包后的结果目录下包含PyQt5文件夹,将PyQt5\Qt\plugins下的所有内容(如下图)拷贝到打包结果目录。这样就可以解决PyQt plugins缺失的问题。

2.动态链接库缺失问题

更一般的,打包后可能会缺失某些动态链接库,造成执行程序出错,如

ImportError: DLL load failed: 找不到指定的模块
在打包过程中一般会有与此相关的warning提示(lib not found)无法找到这些动态链接库。例如在32位版本的打包中,可能会出现scipy模块相关的dll文件无法找到。这时就需要在打包的spec文件中指定动态链接库路径,使其关联到打包后的路径中。

binaries=[('C:\\Program Files\\Python36-32\\Lib\\site-packages\\scipy\\extra-dll','.')]
Analysis下的binaries是为打包文件添加二进制文件,缺失的动态链接库可以通过这种方式自动加入到打包路径中。

3.窗体风格变化问题

在某些情况下,如在精简环境下的python程序打包中,执行打包后的程序会出现窗体风格变为老式的win风格,这是由于打包时候PyQt的styles动态库没有找到。因此只需要在Python 目录下找到 Lib\site-packages\PyQt5\Qt\plugins\styles,将styles整个目录复制到打包结果目录。

4.UnicodeDecodeError

当打包时出现类似错误时:

UnicodeDecodeError: 'utf-8' codec can't decode byte 0xce in position 122
可在打包的命令行中输入chcp 65001设置命令行显示utf-8字符,然后再执行打包命令。或者,修改pyinstaller包下的compat.py,根据报错对应的行将

out = out.decode(encoding)
改为

out = out.decode(encoding, 'replace')
冻结打包路径
执行打包后的程序,经常会出现程序使用的图标无法显示,程序使用的关联文件无法关联。或者,在打包的本机上运行正常,但是将打包后的程序放到其它机器上就有问题。这些现象都很有可能是由程序使用的文件路径发生改变产生的,因此在打包时候我们需要根据执行路径进行路径“冻结”。

1.使用绝对路径

在python代码中使用绝对路径调用外部文件可以保证打包时候路径可追溯,因此在本机上运行打包后程序基本没问题。但是当本机上对应路径的资源文件被改变,或者将打包程序应用到别的机器,都会出现搜索不到资源文件的问题。这种方式不是合适的打包发布python软件的方式。

2.使用冻结路径

增加一个py文件,例如叫frozen_dir.py

# -*- coding: utf-8 -*-
"""
Created on Sat Aug 25 22:41:09 2018
frozen dir
@author: yanhua
"""
import sys
import osdef app_path():"""Returns the base application path."""if hasattr(sys, 'frozen'):# Handles PyInstallerreturn os.path.dirname(sys.executable)return os.path.dirname(__file__)

其中的app_path()函数返回一个程序的执行路径,为了方便我们将此文件放在项目文件的根目录,通过这种方式建立了相对路径的关系。

源代码中使用路径时,以app_path()的返回值作为基准路径,其它路径都是其相对路径。以本文中使用的python项目打包为例,如下所示

import frozen_dir
SETUP_DIR = frozen_dir.app_path()FONT_MSYH = matplotlib.font_manager.FontProperties(fname = SETUP_DIR + r'\data\fonts\msyh.ttf',size = 8)DIR_HELP_DOC = SETUP_DIR + r'\data\docs'
DIR_HELP_VIDEO = SETUP_DIR + r'\data\videos'

通过冻结路径,使用了基准目录下的data目录下的fonts, docs, videos。

主程序中也做了类似的调整,改变其设置路径方法

import frozen_dirSETUP_DIR = frozen_dir.app_path()+r'\lib'
sys.path.append(SETUP_DIR)

使用这样的方法进行打包,打包后的可执行程序就可以在其它机器上运行。

其它问题
由于操作系统和运行环境的不同,pyinstaller打包中还可能遇到很多其它问题,最后总结一些我在打包中遇到的其它坑:

1.权限问题

通常时在打包时出现的某些文件拒绝访问或没有权限执行某些操作等。解决这个的方法一般有这几个方面:

a)使用管理员权限运行cmd或其它命令行窗口

b)关闭杀毒软件

c)使用完全权限的管理员账户

2.中文路径

pyinstaller打包后的路径使用中文没有问题,不过为了减少打包时候出错的可能,尽量将打包使用的资源文件和代码文件路径设置为英文。

3.打包后文件的大小

通常python打包为可执行文件都会得到一个较大的包,这是无法避免的,但是我们还是可以通过一些方法来尽量精简打包后的执行程序:

a)在代码中减少不必要的import,如from xxx import *

b)在精简的运行环境(如原生python环境)下打包,缺什么包就下什么包,避免不必要的python包被打包入程序。尤其是anaconda这样的集成环境下打包的结果会大很多。

c)使用UPX

Python 打包工具 pyinstaller相关推荐

  1. 类似pyinstaller_全网首份Python打包工具Pyinstaller实战指南,如丝滑般体验

    写了个吊炸天的Python项目,把我和左手相处的时间都赔上了.但出于版权考虑,我不太想让使用方直接用我的代码,毕竟Python代码给出去,就真的收不回来了.学习Python中有不明白推荐加入交流群 号 ...

  2. python打包工具 --- pyinstaller

    安装 安装python并添加到环境变量之后,在终端执行如下命令即可: pip install pyinstaller 截图如下: 若安装失败,可到: https://www.lfd.uci.edu/~ ...

  3. python 打包 .app 运行 控制台窗口_Python打包工具PyInstaller的安装与pycharm配置支持PyInstaller详细方法...

    windows系统下安装Pyinstaller cmd下输入指令 pip install PyInstaller Pyinstaller的使用 进入需要打包的目录下,执行打包命令 Pyinstalle ...

  4. python打包工具报错_Python打包发布神器—Pyinstaller

    一前言 在windows平台学习python的过程中,你肯定会遇到需要把.py脚本打包成.exe的情形,如此,至少有两方面的好处:第一,你的代码保密性更好,其他人不能直接看到python代码:第二,打 ...

  5. Python打包工具Pyintealler打包py文件为windows exe文件过程及踩坑记录+实战例子

    Python打包工具Pyintealler打包py文件为windows exe文件过程及踩坑记录+实战例子 目录 Python打包工具Pyintealler打包py文件为windows exe文件过程 ...

  6. python打包工具哪个好用_python打包工具比较

    前一段用python写了点小工具,希望能给同事用,这里总结一下python的打包以及构建的方法. 首先是一些需要安装依赖包的方法,这也是比较推荐的正统的方法. 1.setuptools or pip ...

  7. Python打包工具

    目前几个主要的活跃 PyInstaller,成熟,用户最多:其原理就是复制文件.支持打包成onefile的可执行文件.目录两个方式.PyArmor有对其官方适配. cx_Freezer,原理与PyIn ...

  8. Python打包工具setuptools的使用

    将我们写的Python程序发布成包后,可以使其能够安装使用. 在项目上测试的时候,某些情况下,可以将Python打包,然后上传到测试服务器,安装测试. setuptools是常用的打包工具. 一个简单 ...

  9. Setuptools(Python打包工具)

    目录 前言 一.安装 二.源码包介绍 2-1.源码包sdist 2-1-1.打包成源码包sdist 2-1-2.安装源码包 2-2.二进制包bdist 2-2-1.打包成源码包bdist 2-2-2. ...

最新文章

  1. Linux下的网络远程安装
  2. 多线程:什么是ThreadLocal?应用场景?
  3. major头文件_JPEG头文件结构及组成
  4. Kettle使用_2变量替换之设置标量
  5. linux下qt生成可安装的程序,linux – 如何为Qt应用程序创建“安装”包?
  6. Mr.J--Java异常处理总结
  7. TreeSet学习,比较器学习
  8. 【java笔记】线程(5):线程安全问题
  9. 《Flutter 从0到1构建大前端应用》读后感—第5章【动画】
  10. hibernate教程笔记5
  11. Python yolo 神经网络实现视频中的行人车辆检测计数(车流量统计)YOLO多目标跟踪与计数
  12. matlab gui制作,MATLAB GUI制作教程
  13. C. Mortal Kombat Tower(DP)
  14. 一个程序员的真实故事上
  15. Python+AI智能编辑人脸
  16. 大数据解读《旅行青蛙》崛起之谜
  17. JavaScript中字符串连接/拼接的四种方式
  18. iOS 保持界面流畅的技巧,满满都是收获。
  19. 计算机专业的学习方法
  20. UTC时间与Beijing时间转换工具

热门文章

  1. 网络安全-配置kali linux
  2. 在推广初期该如何运营应用
  3. java中的switch用法,循环,方法,数组以及类
  4. java stringformatter_Java 字符串格式化 —— java.util.Formatter
  5. 决战平安京怎么在电脑上玩 决战平安京模拟器教程
  6. 奥特乐成立9个月斩获三轮融资,硬折扣店找到增长新途径?
  7. bilibili视频列表名字太长显示不全的解决方法
  8. 如何设置DevExpress WinForm皮肤样式
  9. 青春如你,不曾消逝。
  10. win10更新卡住不动_安装Win10 2004系统卡在61%|Win10更新卡住解决方法