Python Py与Pyc文件相互编译(保护源码“不实用”)

本文介绍Python源码保护方法,探讨Pyc文件的生成与转换过程,并分享批量转换Pyc文件的脚本,帮助开发者更好地保护自己的Python代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Python 做为一种解释型语言,做为服务端程序还好,但如果做为客户端程序,就有了暴露源码的问题。很多开发者在寻求安全、便捷的发布程序的方法。
  1. Pyc是一种二进制文件,是由py文件经过编译后,生成的文件,是一种byte code,py文件变成pyc文件后,加载的速度有所提高
  2. 而且Pyc是一种跨平台的字节码,是由Python的虚拟机来执行的,这个是类似于JAVA或者.NET的虚拟机的概念
  3. Pyc的内容,是跟python的版本相关的,不同版本编译后的Pyc文件是不同的
  4. Pyc还可以转回Py
单个转换:

Py转Pyc
python xxx.py 其实可以生成.pyc文件

# 命令生成:
python3 -m py_compile /home/demo/test.py   #跟随完整路径(当前文件夹可直接文件名)

# 代码生成:
import py_compile
py_compile.compile('/home/demo/test.py')

Pyc转Py
xxx.pyc : 需要转换的.pyc文件
xxx.py : 转换后文件名

# 命令下载库:
pip install uncompyle6 -i https://pypi.tuna.tsinghua.edu.cn/simple

# 代码生成:
 uncompyle6 -o  xxx.py xxx.pyc

这两种方法生成的.pyc文件都在当前文件夹下的__pycache__文件夹里;
根据Pyc反编译出来的文件有可能会有缩进或逻辑问题(有一定概率)!

批量转换

编译Py文件为Pyc,并且移除编译生成的文件夹和编译前的Py文件

使用说明:
1.将需要编译的项目复制出来,放到compile_dir_and_changename.py所在的文件夹下。(重要:因为编译后会移除Py文件)
2.双击compile_dir_and_changename.py文件,双击后有二次确认;输入Y/y,回车后完成编译

# -*- coding: utf-8 -*-
# @Time    : 2022/12/12 12:13
# @Author  : 微酷
# @Email   : 110169011@qq.com
# @FileName: Py转Pyc.py
# @Software: PyCharmPro2021.3; Anaconda(Python3.10)

import compileall
import os
import shutil

'''
os.getcwd()获取当前工作目录
splitext() 分离文件名和扩展名,返回一个数组。(运行的文件排除)
os.listdir:参数为文件夹路径,可以返回文件夹下的所有子文件夹、文件名称。
os.walk:参数为文件夹路径,返回3个内容:[绝对路径、子文件夹、文件名]。 此方法可以遍历文件夹下的所有文件、子文件及内的所有文件:
	子文件夹如果为空,那么第二个数组为空
    
os.path.isfile(path);
os.path.isdir(path);

复制文件
shutil.copy(文件路径,文件夹路径)
'''

def get_data():
    walk_list = list(os.walk(os.getcwd()))

    # key为当前文件夹下所有的文件夹的路径(包含所在的根目录),value为对应的文件
    data = {}
    for _absolute_path, folder_name, _file_name in walk_list:
        data[_absolute_path] = _file_name
    return data
    
def rename():
    file = get_data()
    for absolute_path, file_names in file.items():
        # 筛选__pycache__文件夹
        if absolute_path.split('\\')[-1] == '__pycache__':
            # 遍历文件夹下的所有文件
            for file_name in file_names:
                # 拆分文件夹名
                file_items = file_name.split('.')
                # 确保是pyc文件
                if file_items[-1] == 'pyc':
                    os.rename(absolute_path + '\\' + file_name,
                              absolute_path + '\\' + file_items[0] + '.' + file_items[-1])


# 复制pyc文件时, 删除多余的py文件和编译生成的__pycache__文件夹
def copy_pyc():
    file = get_data()
    for absolute_path, file_names in file.items():
        # 筛选__pycache__文件夹
        if absolute_path.split('\\')[-1] == '__pycache__':
            # 遍历文件夹下的所有文件
            for file_name in file_names:
                # 拆分文件夹名
                file_items = file_name.split('.')
                # 确保是pyc文件
                if file_items[-1] == 'pyc':
                    # 复制到上级文件夹
                    shutil.copy(absolute_path + f'\\{file_name}', absolute_path + '\\..')
                    os.remove(absolute_path + f'\\{file_name}')
            os.rmdir(absolute_path)

            # 删除.py文件
            # 获取pyc文件夹对应的py文件夹
            path_py = absolute_path[:len(absolute_path)-len('\\__pycache__')]

            # 获取文件夹中的文件列表并遍历
            list_py = file[path_py]
            for py in list_py:
                # 拆分文件夹名
                py_items = py.split('.')
                # 确保是py文件
                if py_items[-1] == 'py':
                    os.remove(path_py + f'\\{py}')

if __name__ == '__main__':
    for name in os.listdir():
        if os.path.isdir(f'.\\{name}') and name[0] != '.':
            print(f'开始编译[.\\{name}]下的项目...')
            confirm = input('此操作会删除py文件,确认请输入Y或y后回车,输入其他均为取消操作:')
            if confirm in ['Y', 'y']:
                compileall.compile_dir(f'.\\{name}')  # 编译当前文件夹中的py文件
                rename()
            else:
                print(f'取消.\\{name}]编译操作')
                break

            # 先复制pyc文件,然后删除.py的文件和__pycache__(根据__pycache__文件夹的上一页)
            copy_pyc()
# -*- coding: utf-8 -*-
# @Time    : 2022/12/12 13:18
# @Author  : 微酷
# @Email   : 110169011@qq.com
# @FileName: 移除__pycache__文件夹的脚本.py
# @Software: PyCharmPro2021.3; Anaconda(Python3.10)

import os
import shutil

def get_data():
    walk_list = list(os.walk(os.getcwd()))

    # key为当前文件夹下所有的文件夹的路径(包含所在的根目录),value为对应的文件
    data = {}
    for _absolute_path, folder_name, _file_name in walk_list:
        data[_absolute_path] = _file_name
    return data


if __name__ == '__main__':
    file = get_data()
    for absolute_path, file_names in file.items():
        # 筛选__pycache__文件夹
        if absolute_path.split('\\')[-1] == '__pycache__':
            # 遍历文件夹下的所有文件
            for file_name in file_names:
                # 拆分文件夹名
                file_items = file_name.split('.')
                # 确保是pyc文件
                if file_items[-1] == 'pyc':
                    os.remove(absolute_path + f'\\{file_name}')
            os.rmdir(absolute_path)

在这里插入图片描述

注意: 如果是开发一个自制包给别人用,恐怕就难以保护源码了(当然可能这样也不符合开源精神吧。这样的场景恐怕只能将源码.py文件转为.pyc文件,像征性地保护一下。因为.pyc文件能够转回源码,并且很容易。虽然会有极少的缺失,大部分是正确的。并且 .pyc 文件无法被 IDE解析到,那么现代IDE的一些智能补全功能就完全用不了,无法查看函数的参数个数、类型、说明,类有哪些方法、成员等等,可以说是体验相当差,都是因为你把自制包中的 .py 转成了 .pyc 文件,导致你的自制包相当难用。

这种情况下,我认为可以用 再包一层的方法解决。你认为重要的代码,转成 .pyc文件,或者写成dll,但暴露给用户的类、函数、接口等切不可转为 .pyc,可以在其中写好一些包装的类、函数,来调用pyc、dll 中的函数!重要保密的文件建议用更好的加密编译方式!

免责声明:本篇文章提供的一切软件、教程和内容信息仅限用于学习和研究目的;不得将上述内容用于商业或者非法用途,否则,一切后果请用户自负。本信息来自网络收集整理,版权争议与本人无关。您必须在下载后的24个小时之内,从您的电脑或手机中彻底删除上述内容。如果您喜欢该程序和内容,请支持正版,购买注册,得到更好的正版服务。我们非常重视版权问题,如有侵权请私信与我联系,敬请谅解!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

WK-良人

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

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

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

打赏作者

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

抵扣说明:

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

余额充值