python转pyd

        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的叶子的博客-CSDN博客

更多三维、二维感知算法和金融量化分析算法请关注“乐乐感知学堂”微信公众号,并将持续进行更新。

  • 8
    点赞
  • 48
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 6
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Coding的叶子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值