Pyqt5 實現多線程文件搜索的案例
我學Java的時候也用Swing做瞭一個文件搜索的小程序,但界面真的挺醜的,現在學瞭點python,感覺python是最簡單的語言,沒有之一。 (大傢沒事都可以來學的,真的很簡單有趣哦)
我采用的是pyqt5,所以需要先安裝Pyqt5模塊
直接cmd命令輸入
pip install pyqt5
閑言少敘,上代碼!!
# -*- coding: utf-8 -*- # @Time : 2018\9\15 20:39 # @Author : Tang weiyang # @File : FileSearch02.py from PyQt5.QtCore import * from PyQt5.QtWidgets import * from PyQt5.QtGui import QIcon import sys import os import threading class fileSearchThread(QThread): sinOut = pyqtSignal(str) # 自定義信號,執行run()函數時,從相關線程發射此信號 def __init__(self,key): super().__init__() self.key = key def run(self): threads=[] path = [r"c:\\", r"d:\\", r"e:\\", r"f:\\"] #通過多線程對windows下的多個盤符進行文件的遍歷查找 for each in path: t = threading.Thread(target=self.search, args=(self.key,each,)) threads.append(t) t.start() for i in range(len(threads)): #將主線程阻塞 threads[i].join() print("搜索結束") def search(self,keyword, path): for dirpath, dirnames, filenames in os.walk(path): for filename in filenames: if filename.__contains__(keyword): print(os.path.join(dirpath, filename)) self.sinOut.emit(os.path.join(dirpath, filename)) for folder in dirnames: if folder.__contains__(keyword): print(os.path.join(dirpath,folder)) self.sinOut.emit(os.path.join(dirpath,folder)) class fileSearch(QListWidget): def __init__(self): super().__init__() self.Ui() def Ui(self): self.key= QLineEdit() self.bt=QPushButton("搜索") self.result = QListWidget() self.bt.clicked.connect(self.ButtonClicked) #按鈕單擊信號綁定到槽 # self.line.editingFinished.connect(self.Action) self.key.editingFinished.connect(self.ButtonClicked) grid = QGridLayout() grid.setSpacing(10) # 創建標簽之間的空間 grid.addWidget(self.key, 1, 0) # (1,0)表示顯示的位置 grid.addWidget(self.bt, 1, 1) grid.addWidget(self.result, 2, 0, 5, 2) # 指定組件的跨行和跨列的大小,指定這個元素跨5行顯示 self.setLayout(grid) for i in range(1,100): self.result.addItem("搜索"+str(i)+"個項目") self.result.itemClicked.connect(self.Clicked) self.setGeometry(300, 300, 500, 500) self.setWindowTitle('文件搜索') self.setWindowIcon(QIcon('icon.jpg')) self.show() def Clicked(self, item): QMessageBox.information(self, "ListWidget", "You clicked: " + item.text()) os.startfile(item.text()) #打開文件 def ButtonClicked(self): # 創建新線程,將自定義信號sinOut連接到slotAdd()槽函數 keyword = self.key.text() self.result.clear() self.thread=fileSearchThread(keyword) self.thread.sinOut.connect(self.slotAdd) self.thread.start() def slotAdd(self,filename): self.result.addItem(str(filename)) if __name__ == '__main__': app = QApplication(sys.argv) ex = fileSearch() sys.exit(app.exec_())
這個小程序可以搜索本地所有的文件,時間大概在10秒左右,點擊文件的地址,就可以打開這個文件.多線程這個有點卡殼,然後寫的很變扭.
這個小程序還有一個小感悟:
涉及到GUI的程序最好要將UI界面和數據處理(還包括大批量文件的讀取,循環計算),否則會導致UI界面無響應,這一點可以參考我的代碼,通過一個多線程很好解決這個問題
線程之間的信息傳遞,可以通過信號和槽完成,
sinOut = pyqtSignal(str) #自定義一個信號槽 self.thread.sinOut.connect(self.slotAdd)#信號綁定到槽 self.sinOut.emit(os.path.join(dirpath,folder)) #發射信號
pyqt的佈局真的很好用,GridLayout真的超級好用
效果圖如下
補充:pyqt5多線程-簡單例子
一、主要代碼邏輯
from PyQt5 import QtWidgets, QtCore from testqt.TEST_QT_FROM import Ui_Dialog import sys from PyQt5.QtCore import * import time # 繼承QThread class Runthread(QtCore.QThread): # python3,pyqt5與之前的版本有些不一樣 # 通過類成員對象定義信號對象 _signal = pyqtSignal(str) def __init__(self): super(Runthread, self).__init__() def __del__(self): self.wait() def run(self): print("run 666") self._signal.emit("run 666"); # 信號發送 class TestQtFromC(QtWidgets.QWidget, Ui_Dialog): text ="" def __init__(self): super(TestQtFromC, self).__init__() self.setupUi(self) #click def timer_click(self): self.thread = Runthread() # 創建線程 self.thread._signal.connect(self.callbacklog) # 連接信號 self.thread.start() # 開始線程 # callback def callbacklog(self, msg): self.text =self.text+time.strftime("%Y-%m-%d %H:%M:%S ", time.localtime())+msg+ "\n" print(self.text) # 回調數據輸出到文本框 self.textEdit.setText(self.text); if __name__ == "__main__": app = QtWidgets.QApplication(sys.argv) mTestQtFromC = TestQtFromC() mTestQtFromC.show() sys.exit(app.exec_())
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- 利用Python制作百度圖片下載器
- Python3中PyQt5簡單實現文件打開及保存
- Python+PyQT5實現手繪圖片生成器
- 使用 PyQt5 設計下載遠程服務器日志文件程序的思路
- PyQt5中QTimer定時器的實例代碼