需求
要写一个数据处理的小程序,对输入的excel文件进行少许处理后从长ID转成宽ID后再保存,用pandas很容易解决这个问题,但要打包成有gui界面的exe文件在windows下使用。
一般能找到的教程是用pyinstaller
,但这样打出来的文件或文件夹巨大无比而且动不动报错。还有用py2exe
, cz_freeze
等方法,但是试了下也不能解决我的问题。
以前在linux系统下有过批量py文件为so以加密py文件的经验,那在windows下,可能也有类似的方法去解决打包需求。
学习网上资料后,准备用cython
把py文件编译成c文件,然后再用visual studio里的cl.exe编译成exe文件。
环境
- Python3, 装在
C:\Anaconda3
- 安装
cython
,pyqt5
,pandas
包- 为了控制最后的压缩包大小,可以
conda create
一个环境再去打包,不过我没有搞
- 为了控制最后的压缩包大小,可以
- vs2017: 装在
C:\Program Files (x86)\Microsoft Visual Studio\2017
-
PATH
按常规,加入相应的路径,但是注意没有设置PYTHONHOME
,PYTHONPATH
等变量。
Cython转换源py文件为.c文件
cython -3 .\longID2wide.py --embed
会生成一个 longID2wide.c
文件
用vs 2017的cl.exe打包成exe文件
cl.exe
要在系统PATH, 当然你写绝对路径也行
cl -I"C:\Anaconda3\include" -I"C:\Program Files (x86)\Windows Kits\10\Include\10.0.17763.0\ucrt" -I"C:\Program Files (x86)\Microsoft Visual Studio\2017\BuildTools\VC\Tools\MSVC\14.16.27023\include" -I"C:\Program Files (x86)\Windows Kits\10\Include\10.0.17763.0\shared" /Tc longID2wide.c /link /OUT:"C:\L2W\l2w.exe" /LIBPATH:"C:\Anaconda3\libs" /LIBPATH:"C:\Program Files (x86)\Microsoft Visual Studio\2017\BuildTools\VC\Tools\MSVC\14.16.27023\lib\x64" /LIBPATH:"C:\Program Files (x86)\Windows Kits\10\Lib\10.0.17763.0\um\x64" /LIBPATH:"C:\Program Files (x86)\Windows Kits\10\Lib\10.0.17763.0\ucrt\x64" /ENTRY:"wmainCRTStartup" /MACHINE:X64 /SUBSYSTEM:windows
注意点(重要!!):
-
-I
后是.h
头文件的目录,/LIBPATH
是.lib
文件的目录,相应要放入到系统变量PATH,或者如同我一样在编译时指定。 - 建议装个
everything
,如果系统提示你缺少xxxxx.h
或xxxx.lib
,去vs2017
和python
的目录,以及其他可能的相关目录下搜索。- 上面的命令就是提示少什么文件,然后我用
everything
去找再加入到命令行中去
- 上面的命令就是提示少什么文件,然后我用
- 生成的exe对像为
/link /OUT:"C:\L2W\l2w.exe"
, 260k左右大小
测试
此时去运行这个文件,很有可能会出现如下问题
Fatal Python error: Py_Initialize: Unable to get the locale encoding
ImportError: No module named 'encodings'
如何解决?按照网上的做法,在系统里设置PATHONHOME
,指向C:\Anaconda3
目录,程序能正常工作。
打包
但是,如上,把这个exe文件发给一个没有python环境的人,仍然不能运行?这是什么原因?
原因是,python本身是一个解释型的语言,其运行要依赖于在系统里安装的python基础包和安装上的包,或是动态链接库。在没有PYTHONHOME
情况下,自然不能运行。
那如何解决?仔细一想,其实大部分win32程序找动态依赖库,首先是从当前目录下找,然后再去系统路径里找,因此把这个exe文件放入到一个有python环境的目录
就有可能解决问题。
因此,我把C:\Anaconda3\envs\py37
目录下的东西整个复制到C:\L2W
,此exe文件就可正常运行!!
精减
这个C:\py3
里面还有多文件,能不能删除?经过我实践后发现根目录下,仅需要保留python37.dll
文件,DLLs
和Lib
文件夹,就可正常运行。
其实上述文件夹下还可以进一步删除文件,但是搞不好删掉一个文件或文件夹,编译而成的exe文件无法运行。而且,实现我代码功能的pandas
,nunpy
, pyqt5
包本身就很大(50m,50m,130m),删除一些周边的小文件或文件夹对精减没有太大意义,因此就不用花太多时间去测试极限运行依赖性。
有点卸磨杀驴
的是, Lib/site-packages
下的Cython
文件夹是可以删除的,毕竟运行时用不到编译工具。
网友评论