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。如有錯誤或未考慮完全的地方,望不吝賜教。

推薦閱讀: