Python如何實(shí)現(xiàn)按文件夾加密解密其中文件
Python實(shí)現(xiàn)按文件夾加密解密其中文件工具
一個(gè)使用Python實(shí)現(xiàn)的按文件夾加密解密其中文件的程序,包含GUI界面和較為完善的功能。我們將使用PyQt6來(lái)創(chuàng)建GUI,使用cryptography庫(kù)進(jìn)行加密操作。該程序用到一些模塊/庫(kù):
PyQt6 庫(kù)是第三方庫(kù),需要安裝,用于創(chuàng)建圖形用戶界面(GUI)的庫(kù)。PyQt6的簡(jiǎn)介與安裝,可參見Pyside6 安裝和簡(jiǎn)單界面開發(fā)過(guò)程詳細(xì)介紹
cryptography 是第三方庫(kù),需要安裝,用于加密和解密的庫(kù)。
sys、os 和 base64是Python自帶的標(biāo)準(zhǔn)庫(kù)模塊,不需要安裝。
運(yùn)行效果
源碼如下:
import sys import os from PyQt6.QtWidgets import (QApplication, QWidget, QPushButton, QVBoxLayout, QHBoxLayout, QFileDialog, QLineEdit, QLabel, QMessageBox, QProgressBar, QTextEdit, QSplitter) from PyQt6.QtCore import Qt, QThread, pyqtSignal from PyQt6.QtGui import QFont, QColor from cryptography.fernet import Fernet from cryptography.hazmat.primitives import hashes from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC import base64 class CryptoWorker(QThread): progress = pyqtSignal(int) finished = pyqtSignal() error = pyqtSignal(str) status = pyqtSignal(str) # 新增狀態(tài)信號(hào) def __init__(self, folder, password, mode): super().__init__() self.folder = folder self.password = password self.mode = mode # 'encrypt' or 'decrypt' def run(self): try: key = self.get_key(self.password) fernet = Fernet(key) total_files = sum([len(files) for r, d, files in os.walk(self.folder)]) processed_files = 0 self.status.emit(f"開始{'加密' if self.mode == 'encrypt' else '解密'}操作...\n") for root, dirs, files in os.walk(self.folder): for file in files: file_path = os.path.join(root, file) relative_path = os.path.relpath(file_path, self.folder) if self.mode == 'encrypt': if file.endswith('.enc'): self.status.emit(f"跳過(guò)已加密文件: {relative_path}") else: self.encrypt_file(fernet, file_path) self.status.emit(f"已加密: {relative_path}") else: # decrypt if not file.endswith('.enc'): self.status.emit(f"跳過(guò)未加密文件: {relative_path}") else: self.decrypt_file(fernet, file_path) self.status.emit(f"已解密: {relative_path}") processed_files += 1 self.progress.emit(int(processed_files / total_files * 100)) self.status.emit("\n操作完成!") self.finished.emit() except Exception as e: self.error.emit(str(e)) def get_key(self, password): password = password.encode() salt = b'salt_' kdf = PBKDF2HMAC( algorithm=hashes.SHA256(), length=32, salt=salt, iterations=100000, ) key = base64.urlsafe_b64encode(kdf.derive(password)) return key def encrypt_file(self, fernet, file_path): with open(file_path, 'rb') as f: data = f.read() encrypted = fernet.encrypt(data) encrypted_path = file_path + '.enc' with open(encrypted_path, 'wb') as f: f.write(encrypted) os.remove(file_path) def decrypt_file(self, fernet, file_path): with open(file_path, 'rb') as f: data = f.read() try: decrypted = fernet.decrypt(data) decrypted_path = file_path[:-4] # 移除 '.enc' with open(decrypted_path, 'wb') as f: f.write(decrypted) os.remove(file_path) except: raise Exception(f'解密 {file_path} 失敗,可能密碼錯(cuò)誤') class Encryptor(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): self.setWindowTitle('文件夾加密器') self.setGeometry(300, 200, 700, 600) main_layout = QVBoxLayout() # 上半部分:控制面板 control_panel = QWidget() control_layout = QVBoxLayout() # 文件夾選擇 folder_layout = QHBoxLayout() self.folder_input = QLineEdit() folder_button = QPushButton('選擇文件夾') folder_button.clicked.connect(self.select_folder) folder_layout.addWidget(self.folder_input) folder_layout.addWidget(folder_button) control_layout.addLayout(folder_layout) # 密碼輸入 password_layout = QHBoxLayout() password_label = QLabel('密碼:') self.password_input = QLineEdit() self.password_input.setEchoMode(QLineEdit.EchoMode.Password) password_layout.addWidget(password_label) password_layout.addWidget(self.password_input) control_layout.addLayout(password_layout) # 進(jìn)度條 self.progress_bar = QProgressBar() control_layout.addWidget(self.progress_bar) # 加密解密按鈕 button_layout = QHBoxLayout() encrypt_button = QPushButton('加密') decrypt_button = QPushButton('解密') encrypt_button.clicked.connect(lambda: self.process_folder('encrypt')) decrypt_button.clicked.connect(lambda: self.process_folder('decrypt')) button_layout.addWidget(encrypt_button) button_layout.addWidget(decrypt_button) control_layout.addLayout(button_layout) control_panel.setLayout(control_layout) main_layout.addWidget(control_panel) # 分割線 splitter = QSplitter(Qt.Orientation.Vertical) # 注意事項(xiàng)面板 notice_panel = QWidget() notice_layout = QVBoxLayout() notice_label = QLabel("注意事項(xiàng):") notice_label.setFont(QFont("Arial", 12, QFont.Weight.Bold)) notice_label.setStyleSheet("color: red;") notice_text = QTextEdit() notice_text.setPlainText( "1. 請(qǐng)確保記住密碼,否則將無(wú)法解密文件。\n" "2. 請(qǐng)慎重操作!該程序會(huì)直接在原文件夾中操作,謹(jǐn)防誤操作。\n" "3. 加密后刪除原文件,解密后刪除加密文件。加密后的文件會(huì)有 .enc 擴(kuò)展名。\n" "4. 本程序支持加密/解密文件夾中的所有文件。\n" "5. 已加密的文件不會(huì)被重復(fù)加密,未加密的文件不會(huì)被解密。" ) notice_text.setReadOnly(True) notice_text.setFont(QFont("Arial", 11)) notice_text.setStyleSheet("color: #D32F2F;") # 深紅色 notice_layout.addWidget(notice_label) notice_layout.addWidget(notice_text) notice_panel.setLayout(notice_layout) # 日志面板 log_panel = QWidget() log_layout = QVBoxLayout() log_label = QLabel("處理日志:") #log_label.setFont(QFont("Arial", 10, QFont.Weight.Bold)) self.log_text = QTextEdit() self.log_text.setReadOnly(True) #self.log_text.setFont(QFont("Arial", 10)) log_layout.addWidget(log_label) log_layout.addWidget(self.log_text) log_panel.setLayout(log_layout) # 添加面板到分割器 splitter.addWidget(notice_panel) splitter.addWidget(log_panel) main_layout.addWidget(splitter) self.setLayout(main_layout) def select_folder(self): folder = QFileDialog.getExistingDirectory(self, "選擇文件夾") self.folder_input.setText(folder) def process_folder(self, mode): folder = self.folder_input.text() password = self.password_input.text() if not folder or not password: QMessageBox.warning(self, '警告', '請(qǐng)選擇文件夾并輸入密碼') return # 清空日志 self.log_text.clear() self.worker = CryptoWorker(folder, password, mode) self.worker.progress.connect(self.update_progress) self.worker.finished.connect(self.process_finished) self.worker.error.connect(self.process_error) self.worker.status.connect(self.update_log) # 連接狀態(tài)信號(hào) self.worker.start() def update_progress(self, value): self.progress_bar.setValue(value) def update_log(self, message): self.log_text.append(message) def process_finished(self): QMessageBox.information(self, '成功', '操作完成') self.progress_bar.setValue(0) def process_error(self, error_msg): QMessageBox.critical(self, '錯(cuò)誤', f'發(fā)生錯(cuò)誤:{error_msg}') self.progress_bar.setValue(0) self.log_text.append(f"\n錯(cuò)誤: {error_msg}") if __name__ == '__main__': app = QApplication(sys.argv) ex = Encryptor() ex.show() sys.exit(app.exec())
到此這篇關(guān)于Python如何實(shí)現(xiàn)按文件夾加密解密其中文件的文章就介紹到這了,更多相關(guān)Python加密解密文件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python opencv鼠標(biāo)畫矩形框之cv2.rectangle()函數(shù)
鼠標(biāo)操作屬于用戶接口設(shè)計(jì),以前一直使用Qt來(lái)做,但是如果只需要簡(jiǎn)單的鼠標(biāo),鍵盤操作,直接調(diào)用opencv庫(kù)的函數(shù)也未嘗不可,下面這篇文章主要給大家介紹了關(guān)于python opencv鼠標(biāo)畫矩形框cv2.rectangle()函數(shù)的相關(guān)資料,需要的朋友可以參考下2021-10-10PySide(PyQt)使用QPropertyAnimation制作動(dòng)態(tài)界面的示例代碼
文章介紹了如何使用PySide或PyQt的QPropertyAnimation類來(lái)創(chuàng)建動(dòng)態(tài)界面效果,感興趣的朋友一起看看吧2025-03-03python執(zhí)行數(shù)據(jù)庫(kù)的查詢操作實(shí)例講解
在本篇文章里小編給大家整理了一篇關(guān)于python執(zhí)行數(shù)據(jù)庫(kù)的查詢操作實(shí)例講解內(nèi)容,有需要的朋友們可以參考學(xué)習(xí)下。2021-10-10Python 關(guān)于模塊和加載模塊的實(shí)現(xiàn)
這篇文章主要介紹了Python 關(guān)于模塊和加載模塊的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03Python自動(dòng)化測(cè)試中yaml文件讀取操作
這篇文章主要介紹了Python自動(dòng)化測(cè)試中yaml文件讀取操作,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08Python使用Windows API創(chuàng)建窗口示例【基于win32gui模塊】
這篇文章主要介紹了Python使用Windows API創(chuàng)建窗口操作,結(jié)合實(shí)例形式分析了Python基于win32gui模塊調(diào)用Windows API創(chuàng)建窗口具體操作步驟與相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2018-05-05