python转pyd的主要目的是为了对源码进行加密,通常运行python文件在__pycache__产生的pyc文件也可以作为加密文件。但是,pyc文件比较容易被反编译出源码。采用pyd加密的方式更加安全,不易于被反编译。
python文件转pyd格式相当于编译成DLL或so文件,即把文件编译成库,然后供其他函数调用。pyd文件可以直接通过import进行调用,使用方法与.py文件一样,例如test.pyd和test.py使用方法保持一致。但是,pyd文件不能直接运行,必须通过import来运行。
1 环境准备
如果python环境中没有Cython,可以通过pip直接进行安装。
pip install Cython
2 单个python文件转换
单个文件转换需要新建setup.py文件,其代码内容如下所示。其运行方式为:
python setup.py build_ext --inplace appname pypath
其中,python setup.py build_ext --inplace是常规的编译命令,appname和pypath是为方便批量调用是引入的可变参数。appname表示打包的名称,可以任意命名,对编译结果没有影响。pypath表示需要打包的python文件路径(xxx.py),例如a/test.py。
示例:
python setup.py build_ext --inplace testapp a_dir/b_dir/c_file.py
运行成功,会在文件夹下生成build文件,并且在当前目录下生成xxx.cyxx-win_xxx.pyd文件,这个pyd文件就是我们所需要的。
import os
import sys
import shutil
from distutils.core import setup
from Cython.Build import cythonize
from distutils.command.build_ext import build_ext
def get_export_symbols_fixed(self, ext):
pass # return [] also does the job!
# replace wrong version with the fixed:
build_ext.get_export_symbols = get_export_symbols_fixed
if __name__ == '__main__':
if os.path.exists('build'):
shutil.rmtree('build')
# print('argv: ', sys.argv, len(sys.argv))
appname = sys.argv[3]
pypath = sys.argv[4]
sys.argv = sys.argv[:3]
# print('argv: ', sys.argv, len(sys.argv))
cpath = pypath.replace('.py', '.c')
if os.path.exists(cpath):
os.remove(cpath)
setup(
name=appname,
ext_modules=cythonize(pypath)
)
if os.path.exists(cpath):
os.remove(cpath)
3 pyd批量转换
pyd批量转换程序如下所示。通过proj_dir指定转换的python文件路径,转换完成后所有文件被保存到respyd_dir目录下。respyd_dir与原始proj_dir目录结构和文件完全一致,区别在于py文件全部被替换成pyd文件。
import os
import glob
import shutil
from pathlib import Path
if __name__ == '__main__':
proj_name = 'py2pyd'
proj_dir = r'py'
proj_dir = str(Path(proj_dir)) + '/'
respyd_dir = r'respyd'
pyd_suffix = '.cp37-win_amd64.pyd'
if os.path.exists(respyd_dir):
shutil.rmtree(respyd_dir)
shutil.copytree(proj_dir, respyd_dir)
py_files = glob.glob(proj_dir + '**', recursive=True)
i = 1
for py in py_files:
py_ = os.path.splitext(py)
filename = os.path.basename(py).rsplit('.', 1)[0]
dirname = os.path.dirname(py)
if py_[-1] != '.py':
continue
print(str(i).zfill(3), ': building', py)
command = 'python setup.py build_ext --inplace ' + proj_name + ' ' + py
os.system(command)
if os.path.exists(filename + pyd_suffix):
dst = dirname.replace(str(Path(proj_dir)), respyd_dir) + '/' + filename + '.pyd'
shutil.copy(filename + pyd_suffix, dst)
os.remove(filename + pyd_suffix)
if os.path.exists(dirname.replace(str(Path(proj_dir)), respyd_dir) + '/' + filename +'.py'):
os.remove(dirname.replace(str(Path(proj_dir)), respyd_dir) + '/' + filename +'.py')
i += 1
# break
4 编译环境问题
如果编译过程中,遇到编译环境报错,可参考博文:python windows编译问题总结_Coding的叶子的博客-优快云博客。
更多三维、二维感知算法和金融量化分析算法请关注“乐乐感知学堂”微信公众号,并将持续进行更新。