定制文件制作与使用

一、制作

'''
说明:
    Project.data6文件     存储项目数据的二进制文件
    info.Bint文件     项目文件中每一个子文件的二进制数据,在Project.data6二进制文件中的起始位置信息,如果缺少该文件,则程序
无法解压Project.data6文件
'''
import sys, os,zipfile
import zipfile as zip
from PyQt5.QtWidgets import *
class myWin(QWidget):
    def __init__(self, parent=None):
        super(myWin, self).__init__(parent)
        self.setWindowTitle('定制格式文件')

        self.setUI()
    def setUI(self):
        lay=QHBoxLayout()
        self.setLayout(lay)
        bt1=QPushButton('保存定制格式文件')
        bt1.clicked.connect(self.WriteFileFormat)
        bt2=QPushButton('解压定制格式文件')
        bt2.clicked.connect(self.ReadFileFormat)
        lay.addWidget(bt1)
        lay.addWidget(bt2)
    def WriteFileFormat(self):
        position_list= []
        position_num=0
        file_name_list=[]
        read_path,_=QFileDialog.getOpenFileNames(self,'选择文件','','所有文件(*)')
        # print(type(read_path))
        # print(read_path)

        for file in read_path:
            # print(file)
            file_name=file.split(sep='/')[-1]
            # print(file_name)
            file_name_list.append(file_name)
            path1=file
            # print(path1)
            with open(path1,'rb') as f1:
                data=f1.read()
                lenght=len(data)
                position_num=position_num+lenght
                position_list.append(position_num)
                # print(path1)
                # print(lenght)
                # print('-'*60)
            with open('Project.data6','ab') as f2:
                f2.write(data)
        # print(position_list)
        start_end_position_list=[] # 打包进文件中的文件,在定制格式文件中数据中的切片位置列表,每一个元素都是该文件的切片位置
        for i in range(len(position_list)):
            if i ==0:
                start_end_position=[0,position_list[i]]
                start_end_position_list.append(start_end_position)
            else:
                start_end_position = [position_list[i-1], position_list[i]]
                start_end_position_list.append(start_end_position)

        # print(file_name_list)
        # print(start_end_position_list)
        info_dict={file_name_list[i]:start_end_position_list[i] for i in range(len(file_name_list))}
        with open('info.Bint','w') as info:
            for key,value in info_dict.items():

                info.write(f'{key}:{value}\n')
        # print(info_dict)

        # 添加至压缩包
        save_path, _ = QFileDialog.getSaveFileName(self, '保存文件', '', '定制文件格式(*.BLH9)')
        zp = zip.ZipFile(file=save_path, mode='w', compression=zipfile.ZIP_STORED,
                         allowZip64=False)
        zp.write(filename='Project.data6', arcname='Project.data6')
        zp.write(filename='info.Bint', arcname='info.Bint')
        os.remove('Project.data6')
        os.remove('info.Bint')
        zp.close()
    def ReadFileFormat(self):
        path,_=QFileDialog.getOpenFileName(self,'打开BHL9文件','','BLH9文件(*.BLH9)')
        zp=zip.ZipFile(file=path,mode='r')
        current_path=os.path.dirname(sys.argv[0])
        zp.extractall(current_path)
        zp.close()
        # read_path,_=QFileDialog.getOpenFileName(self,'打开info文件','','info文件(*.txt1)')
        out_path=QFileDialog.getExistingDirectory(self,'导出BLH9文件','')
        with open(os.path.join(current_path,'info.Bint'),'r') as f1:
            info_data=f1.read()
            # os.remove(os.path.join(current_path,'info.Bint'))
            # print(data.split(sep='\n'))
        with open(os.path.join(current_path,'Project.data6'),'rb') as f2:
            project_data=f2.read()
            # os.remove(os.path.join(current_path,'Project.data6'))
        for info in info_data.split(sep='\n'):
            if len(info) !=0:
                # print(info)
                file_name=info.split(sep=':')[0]
                part1=info.split(sep=':')[1]
                part2=part1.replace('[','')
                part3=part2.replace(']','')
                # print(file_name,'\n',part3,type(part3))
                start_num=int(part3.split(sep=',')[0])
                end_num = int(part3.split(sep=',')[1])
                # print(file_name,start_num,end_num)
                child_data=project_data[start_num:end_num]
                with open(os.path.join(out_path,f'new_{file_name}'),'wb') as f3:
                    f3.write(child_data)

if __name__ == '__main__':
    app = QApplication(sys.argv)
    window = myWin()
    window.show()
    sys.exit(app.exec())

二、使用

1.源码

import win32api,win32con
import os,sys

def SetRegedit(FileSuffix,AppFile):
    # 创建注册表项名称为文件后缀并设置值
    key1=win32api.RegCreateKey(win32con.HKEY_CLASSES_ROOT,FileSuffix)
    win32api.RegSetValueEx(key1, None, 0, win32con.REG_SZ, AppFile)
    # 创建文件名称子项并设置值
    key2=win32api.RegCreateKey(win32con.HKEY_CLASSES_ROOT,AppFile)
    win32api.RegSetValueEx(key2,None,0,win32con.REG_SZ,'TEST小白数据')
    # 设置icon
    path = os.path.dirname(sys.argv[0])
    ico_path=os.path.join(path,'empty.ico')
    key2_icon=win32api.RegCreateKey(win32con.HKEY_CLASSES_ROOT,rf'{AppFile}\DefaultIcon')
    win32api.RegSetValueEx(key2_icon,None,0,win32con.REG_SZ,ico_path)
    # 设置默认打开的exe程序
    key2_exe=win32api.RegCreateKey(win32con.HKEY_CLASSES_ROOT,rf'{AppFile}\Shell\Open\Command')
    # path='C:\Youdao\定制文件阅读器.exe' # 输入exe文件所在的文件路径即可
    path='D:\我的文档\Desktop\制定文件阅读器\定制文件阅读器.exe' #将exe文件放置在D盘同放置在C盘结果一样可行
    win32api.RegSetValueEx(key2_exe,None,0,win32con.REG_SZ, f'{path} "%1"')
    '''
    windows系统中设置的广联达GTJ2025默认打开软件的注册表设置的值为:
        C:\Program Files\Glodon\GTJ2025\bin\GTJ2025.exe "%1"
    仔细观察可以发现两点:
        1.exe后面的%1放置在双引号中;
        2.exe和后面的"%1"之间有一个空格。
    因此,在win32api.RegSetValueEx()的参数f'{path} "%1"'设置时,采用三个技巧:
        1.为了让python解释器区分字符串中的双引号,在整个字符串外面放置了一层单引号;
        2.为防止输入错误,在字符串中引入了path变量;
        3.%1放置在双引号之中;
        4.{path}和"%1"之间添加了空格;
    经测试发现这个方法是可行的,打开注册表后发现,BLH9格式文件的打开程序路径已经设置为:
        C:\Youdao\定制文件阅读器.exe "%1"    
    该路径设置格式同广联达的设置。
    '''
SetRegedit(FileSuffix='.BLH9',AppFile='blh9_flie')

2.打包后的exe文件及配置资源压缩包,见本篇文章资源

运行setWinReg.exe时,切记以管理员身份运行,否则会报错。

三、pywin32

 练习代码,经测试皆可行,若报错,以管理员身份运行pycharm后点击运行程序即可:

import win32api
import win32con
import winreg

name=win32api.GetComputerName()
print(name)
v=win32api.GetVersion()
print(v)
def OpenKeyItem(key,sub_key):
    key = win32api.RegOpenKey(key, sub_key)
    print(key,type(key))
def CreateKeyItem(key,sub_key): # 创建注册表项
    key=win32api.RegCreateKey(key, sub_key)
    print(key,type(key))
    '''
    win32api.RegCreateKeyEx() 未测试成功,总是遇到报错信息:
    AttributeError: module 'win32con' has no attribute 'REG_OPTION_NON_VOLATILE'
    '''
def DeleteKeyItem(key,sub_key): # 删除注册表项
    win32api.RegDeleteKey(key, sub_key)
# OpenKeyItem(key=win32con.HKEY_CURRENT_USER,sub_key=r'bai985')
# CreateKeyItem(key=win32con.HKEY_CURRENT_USER,sub_key=r'baihe985')
# CreateKeyItem(key=win32con.HKEY_CURRENT_CONFIG,sub_key=r'bai818')
# DeleteKeyItem(key=win32con.HKEY_CURRENT_USER,sub_key=r'baihe985')
def QueryKeyValue(key,sub_key,name=None): # 查询注册表的值

    key = win32api.RegOpenKey(key, sub_key)
    value, type = win32api.RegQueryValueEx(key, name)
    print(value,type)
    '''
    win32api.RegQueryValueEx 的参数用法如下:
    hKey:要查询的根键。
    lpValueName:要查询的值的名称。如果没有指定具体的 lpValueName 参数,它将查询子键的默认值。
    lpReserved:保留参数,通常为 0。
    lpType:用于接收值的数据类型。
    lpData:用于接收值的数据。
    lpcbData:用于接收值的数据长度。
    '''
# QueryKeyValue(key=win32con.HKEY_CURRENT_USER,sub_key=r'AppEvents\EventLabels\.Default',name='DispFileName')
def SetKeyValue(key,ValueName=None,ValueType=win32con.REG_SZ,Value=''):
    reg_path = r'TestKey1'
    # 创建注册表键
    # 要创建的值的名称和数据
    value_name = 'TestValue'
    value_data = 'Hello, Registry!'

    # 设置注册表值
    win32api.RegSetValueEx(key,ValueName,0,ValueType,Value)
    '''
    win32api.RegSetValueEx 的参数如下:
    hKey:要操作的根键。
    lpValueName:值的名称,是一个注册表项对象。如果没有指定具体的 lpValueName 参数,它将查询子键的默认值
    reserved:保留参数,通常为 0。
    dwType:值的数据类型,参数值有以下几种:
        win32con.REG_SZ:字符串类型。
        win32con.REG_DWORD:双字(32 位整数)类型。
        win32con.REG_BINARY:二进制类型。
        win32con.REG_EXPAND_SZ:可扩展字符串类型。
        win32con.REG_MULTI_SZ:多字符串类型。
    lpData:值的数据。
    '''
    print(f'注册表键 {reg_path} 和值 {value_name} 已创建。')
# key = win32api.RegCreateKey(win32con.HKEY_CURRENT_USER, r'TestKey1')
# SetKeyValue(key=key,ValueName='TestValue2',ValueType=win32con.REG_SZ,Value='Hello, World!')
# SetKeyValue(key=key,Value='Hello, World!')
# HKEY_CURRENT_USER\AppEvents\EventLabels\.Default
def DeleteKeyValue_winreg(winreg_key, winreg_key_path, winreg_value_name):
    key = winreg.OpenKey(winreg_key, winreg_key_path, 0, winreg.KEY_WRITE)
    winreg.DeleteValue(key, winreg_value_name)
    '''
    使用pywin32删除注册表的值报错,因为没有权限,但是可以用winreg库实现该功能。
    '''
# DeleteKeyValue(winreg_key=winreg.HKEY_CURRENT_USER,winreg_key_path=r"TestKey1",winreg_value_name="TestValue2")
def DeleteKeyValue(key,sub_key,value_name):
    hkey = win32api.RegOpenKey(key, sub_key, 0, win32con.KEY_ALL_ACCESS)
    win32api.RegDeleteValue(hkey, value_name)
    pass
# DeleteKeyValue(key=win32con.HKEY_CURRENT_USER,sub_key=r"TestKey1",value_name="TestValue")

QueryKeyValue(key=win32con.HKEY_CLASSES_ROOT,sub_key=r'MyEtcFile\Shell\Open\Command')

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值