一、制作
'''
说明:
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')