Python .py生成.pyd文件并打包.exe 的注意事项说明


Posted in Python onMarch 04, 2021

最近用python写了一个小程序,想发布出去让人试用又不想暴露源码,搜索了一下发现将py文件编译成pyd文件就能达到目的。

转换过程很简单,但是在调用pyd文件并且打包为单个exe文件的时候遇到一个坑,搞了一天才解决,在这里分享一下。

首先安装cython库

个人比较喜欢用清华的镜像库,速度快。

pip install Cyphton -i https://pypi.tuna.tsinghua.edu.cn/simple

然后创建一个setup.py文件

写入以下内容:

from distutils.core import setup
from Cython.Build import cythonize
setup(ext_modules=cythonize("BetaV14.py"))

BetaV14.py就是要转换为pyd文件的代码文件

命令行输入:

python setup.py build_ext --inplace

会在.py文件目录下生成一个BetaV14.cp37-win_amd64.pyd文件,文件名中“.cp37-win_amd64”这一段可以删除,不删除也可以正常调用;但原文件名字段不能改变。

接下来需要打包发布为.exe文件

我用的是pyinstaller,还是用清华镜像库安装。

pip install pyinstaller -i https://pypi.tuna.tsinghua.edu.cn/simple

根据一些教程,有的说在命令行直接输入:

pyinstaller -F BetaV14.py

就能直接引用pyd文件打包发布exe文件,但是在我这里出现文件缺失的错误:

ValueError: Module file F:\python项目1\BetaV14.py is missing

继续查找问题,发现需要用一个入口程序来导入pyd文件,于是创建一个main.py文件,import刚才生成的模块,pyd文件默认优先级高于py文件,可以在后面解包exe文件来验证。

import BetaV14
if __name__ == '__main__':
 BetaV14()

这里需要注意的是程序的__main__入口只能有一个,如果源py文件中有定义main入口,需要注释掉并调整代码缩进,否则通过main.py调用pyd文件遇到if name == ‘main':之后的代码都不会运行。

接着命令行输入:

pyinstaller -F main.py

打包成.exe文件,在dist目录下发现main.exe文件大小只有5M,之前采用py文件打包的程序有接近50M,运行之后闪退。这个问题想了半天才想出来,可能是引用了大量的第三方库没有打包进去,于是将源py文件头部import部分全部复制到main.py文件头部。

import win32gui
import win32api
import win32con
import time 
import random
import datetime
import os,sys
import configparser
import numpy as np
from PIL import Image
from scipy.signal import convolve2d
import http.client
import subprocess
import BetaV14
if __name__ == '__main__':
 BetaV14()

再次用命令pyinstaller -F main.py打包,得到正常大小的.exe文件,点击能正常运行。

接下来我们用pyinstxtractor.py(不清楚该脚本是否涉及著作权,请自行搜索)解包exe文件验证一下,命令行输入:

python pyinstxtractor.py main.exe

会得到一个main.exe_extracted文件夹,在文件夹下发现文件BetaV14.pyd,说明通过引用pyd文件打包成功。

Python .py生成.pyd文件并打包.exe 的注意事项说明

在此作为一个初学者记录一下自己遇到的坑,让大佬们见笑了。

补充:python打包编译成pyd或者_python之setup.py的那些事

Python .py生成.pyd文件并打包.exe 的注意事项说明

今天偶然对setup.py产生了兴趣,以前只知道可以用它来安装包,例如

python setup.py build ->python setup.py install.当然前提你下载的这个源码包是压缩的,之前对这个理解并不深,今天偶然看见pip install -e . 的用法,然后串起来想了一下。

我的目录结构如上,首先我创建了一个setuptutorial的directory,然后我在下面创建了greet_pkg的python package,并且在setuptutorial下面创建了setup.py如下

from setuptools import setup, find_packages
 
setup(
 name='greet',
 version='1.0.0',
 packages=find_packages(include=['greet_pkg', 'greet_pkg.*']),
 url='',
 license='uestc',
 author='jack',
 author_email='2444093230@qq.com',
 description='test package',
 py_modules=['greet2'],
 install_requires=['pyjokes']
)

greet2.py如下

def greet2(name):
 print(
 'hello',name,'this is greet2'
 )

在greet_pkg下面下了一个greet.py如下

import pyjokes
def greet(name):
 print('hello!', name, f'im telling you a joke {pyjokes.get_joke()}')

整体目录结构和setup.py就如上所示

接下来好戏开场了,如果我要在任意其他文件里面使用到我定义的greet()方法,以前的做法是按照import规则在其他文件里面导入,当然如果写的不规范,及其的容易出问题,这里我提供另外一个思路,在setuptutorial下面使用pip install -e . 命令,将setup.py里面包含的package和py_module安装到Libary root下,当然他的实际的location不是在Libary root下,这个你可以在pip install -e . 之后使用pip show greet 查看他的信息.

到这里就完了吗?

当然没有,这个就是之前的python setup.py build 的作用,我这里猜测大概率是把tar.gz包转化成我上述的目录结构一样的directory。

而python setup.py install 的作用就类似于pip install '-e' . 。而且python setup.py install 之后的greet包是真的存在于sitepackages里面的。

setup.py除了上述安装包的作用,还可以是他的逆过程如 python setup.py sdist 成greet.tar.gz,这样就有上述的装包的过程先build再install。

还可以使用setup.py将py文件转化为pyd,也可以说将pyx文件转化为pyd,

from setuptools import setup
# from distutils.core import setup
from Cython.Build import cythonize
 
# setup(
# name='hello',
# ext_modules=cythonize(['sayhi.py'])
# )

然后运行python setup.py build_ext --inplace就可以了!

pyd文件可以很好的隐蔽py文件里面的实现,和linux下的so文件类似。

有类似py->pyd功能的有easycython模块,可以直接pip安装。

有人可能会说pyc也看不见源码吗?

但是他可以被反编译23333

至于如何将py编译成pyc或pyo

可以使用py_compile或者compileall,不了解的可以自行搜索一下,都有很多的例子,还有针对pyc的反编译库,都可以搜到,至此setup.py我所了解的功能都谈完了,里面还有很多参数可以灵活配置,实现更加炫酷的效果可以查看这个链接setup.py

Python 相关文章推荐
Python实现将DOC文档转换为PDF的方法
Jul 25 Python
Django卸载之后重新安装的方法
Mar 15 Python
详解TensorFlow在windows上安装与简单示例
Mar 05 Python
python使用turtle绘制分形树
Jun 22 Python
Python使用ConfigParser模块操作配置文件的方法
Jun 29 Python
Django中URL的参数传递的实现
Aug 04 Python
Python 取numpy数组的某几行某几列方法
Oct 24 Python
python数据化运营的重要意义
Nov 25 Python
基于h5py的使用及数据封装代码
Dec 26 Python
Django自定义全局403、404、500错误页面的示例代码
Mar 08 Python
python进度条显示之tqmd模块
Aug 22 Python
最新版 Windows10上安装Python 3.8.5的步骤详解
Nov 28 Python
python 中 .py文件 转 .pyd文件的操作
Mar 04 #Python
Python实现图片指定位置加图片水印(附Pyinstaller打包exe)
Mar 04 #Python
python 指定源路径来解决import问题的操作
Mar 04 #Python
python源文件的字符编码知识点详解
Mar 04 #Python
python3判断IP地址的方法
Mar 04 #Python
Python解析m3u8拼接下载mp4视频文件的示例代码
Mar 03 #Python
python和opencv构建运动检测器的实现
Mar 03 #Python
You might like
dedecms防止FCK乱格式化你的代码的修改方法
2007/03/17 PHP
PHP乱码问题,UTF-8乱码常见问题小结
2012/04/09 PHP
PHP重定向的3种方式
2013/03/07 PHP
thinkphp使用literal防止模板标签被解析的方法
2014/11/22 PHP
基于JQuery+PHP编写砸金蛋中奖程序
2015/09/08 PHP
前端必学之PHP语法基础
2016/01/01 PHP
PHP数组遍历的几种常见方式总结
2019/02/15 PHP
javascript 出生日期和身份证判断大全
2008/11/13 Javascript
JAVA四种基本排序方法实例总结
2015/07/24 Javascript
JavaScript中的Object对象学习教程
2016/05/20 Javascript
详解js中call与apply关键字的作用
2016/11/21 Javascript
详解如何在Angular中快速定位DOM元素
2017/05/17 Javascript
基于node打包可执行文件工具_Pkg使用心得分享
2018/01/24 Javascript
vue watch监听对象及对应值的变化详解
2018/02/24 Javascript
Bootstrap Table列宽拖动的方法
2018/08/15 Javascript
Vue.js中provide/inject实现响应式数据更新的方法示例
2019/10/16 Javascript
[07:06]2018DOTA2国际邀请赛寻真——卫冕冠军Team Liquid
2018/08/10 DOTA
python使用urllib2实现发送带cookie的请求
2015/04/28 Python
使用python实现个性化词云的方法
2017/06/16 Python
使用DataFrame删除行和列的实例讲解
2018/04/08 Python
Python图像滤波处理操作示例【基于ImageFilter类】
2019/01/03 Python
OpenCV 模板匹配
2019/07/10 Python
Python高级特性之闭包与装饰器实例详解
2019/11/19 Python
Python实现CAN报文转换工具教程
2020/05/05 Python
详解python定时简单爬取网页新闻存入数据库并发送邮件
2020/11/27 Python
python实现KNN近邻算法
2020/12/30 Python
HTML5中使用postMessage实现两个网页间传递数据
2016/06/22 HTML / CSS
党员自我批评与反省材料
2014/02/10 职场文书
《寓言两则》教学反思
2014/02/27 职场文书
趣味运动会广播稿
2014/09/13 职场文书
幼儿教师师德师风自我剖析材料
2014/09/29 职场文书
通知范文怎么写
2015/04/16 职场文书
2015年度环卫处工作总结
2015/07/24 职场文书
2016年小学感恩节活动总结
2016/04/01 职场文书
学校就业保障协议书
2019/06/24 职场文书
2019如何书写演讲稿?
2019/07/01 职场文书