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 装饰器功能以及函数参数使用介绍
Jan 27 Python
python网络编程学习笔记(七):HTML和XHTML解析(HTMLParser、BeautifulSoup)
Jun 09 Python
全面解读Python Web开发框架Django
Jun 30 Python
python统计日志ip访问数的方法
Jul 06 Python
Python 堆叠柱状图绘制方法
Jul 29 Python
pytorch GAN生成对抗网络实例
Jan 10 Python
浅谈python print(xx, flush = True) 全网最清晰的解释
Feb 21 Python
基于python检查SSL证书到期情况代码实例
Apr 04 Python
Python插件机制实现详解
May 04 Python
Keras实现将两个模型连接到一起
May 23 Python
keras.utils.to_categorical和one hot格式解析
Jul 02 Python
详解Python中@staticmethod和@classmethod区别及使用示例代码
Dec 14 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中的[html][/html]和[code][/code]问题
2007/03/20 PHP
php中$_GET与$_POST过滤sql注入的方法
2014/11/03 PHP
php结合mysql与mysqli扩展处理事务的方法
2016/06/29 PHP
PHP实现绘制二叉树图形显示功能详解【包括二叉搜索树、平衡树及红黑树】
2017/11/16 PHP
CL vs ForZe BO5 第二场 2.13
2021/03/10 DOTA
Javascript 中文字符串处理额外注意事项
2009/11/15 Javascript
深入理解JavaScript定时机制
2010/10/29 Javascript
Extjs TimeField 显示正常时间格式的代码
2011/06/28 Javascript
jquery三个关闭弹出层的小示例
2013/11/05 Javascript
js中的eventType事件及其浏览器支持性介绍
2013/11/29 Javascript
浅谈javascript构造函数与实例化对象
2015/06/22 Javascript
基于jQuery 实现bootstrapValidator下的全局验证
2015/12/07 Javascript
JS控制层作圆周运动的方法
2016/06/20 Javascript
基于Layer+jQuery的自定义弹框
2020/05/26 Javascript
JQuery和PHP结合实现动态进度条上传显示
2016/11/23 Javascript
原生JavaScript实现精美的淘宝轮播图效果示例【附demo源码下载】
2017/05/27 Javascript
在React 组件中使用Echarts的示例代码
2017/11/08 Javascript
浅谈vuex 闲置状态重置方案
2018/01/04 Javascript
解决vue-cli创建项目的loader问题
2018/03/13 Javascript
Vue利用localStorage本地缓存使页面刷新验证码不清零功能的实现
2020/09/04 Javascript
Vue绑定用户接口实现代码示例
2020/11/04 Javascript
[06:07]DOTA2-DPC中国联赛 正赛 Ehome vs VG 选手采访
2021/03/11 DOTA
Python生成随机验证码的两种方法
2015/12/22 Python
Python的爬虫框架scrapy用21行代码写一个爬虫
2017/04/24 Python
一行代码让 Python 的运行速度提高100倍
2018/10/08 Python
Python的高阶函数用法实例分析
2019/04/11 Python
python numpy中cumsum的用法详解
2019/10/17 Python
Python 中 sorted 如何自定义比较逻辑
2021/02/02 Python
党员创先争优公开承诺书
2014/03/28 职场文书
医学生求职自荐书
2014/06/12 职场文书
假释思想汇报范文
2014/10/11 职场文书
2015年质量管理工作总结范文
2015/05/18 职场文书
无违反计划生育证明格式
2015/06/24 职场文书
《灰雀》教学反思
2016/02/19 职场文书
Python语言规范之Pylint的详细用法
2021/06/24 Python
Python Pandas 删除列操作
2022/03/16 Python