信号与槽是pyqt的核心机制,其目的是实现代码与控件之间的交互,真正实现对应的功能,否则控件不会有任何响应。
 信号:是由对象或者控件发射出去的消息,eg:button的单击事件。
 槽:接受事件的信号的代码被成为槽,槽的本质是函数或者方法。
 可以把信号理解为事件,那么槽就是事件函数。
信号与槽的关系:
 信号和槽之间需要绑定,一个信号可以被多个槽拦截,而一个槽也可以接受多个事件的信号。
在Qtdesigner中点击edit选择信号与槽,就可以选中窗体中的控件
  
1. 信号(signal)
 其实就是事件(按钮点击 、内容发生改变 、窗口的关闭事件) 或者是 状态 (check选中了, togglebutton 切换。)
当程序触发了某种状态或者发生了某种事件(比如:按钮被点击了, 内容改变等等),那么即可发射出来一个信号。
2. 槽( slot)
 若想捕获这个信号,然后执行相应的逻辑代码,那么就需要使用到 槽 , 槽实际上是一个函数, 当信号发射出来后,会执行与之绑定的槽函数
3. 将信号与槽链接
 为了能够实现,当点击某个按钮时执行某个逻辑,需要把具体的信号和具体的槽函数绑定到一起.
操作大体流程如下
对象.信号.connect(槽函数)
 案例1:
简单的使用
import sys from PyQt5.QtWidgets import QApplication, QWidget, QPushButton 
class MyWindow(QWidget): def __init__(self): super().__init__() self.init_ui() 
def init_ui(self): # 更改当前Widge的宽高 self.resize(500, 300) # 创建一个按钮 btn = 
QPushButton("点我!", self) # 设置窗口位置、宽高 btn.setGeometry(200, 200, 100, 30) # 
将按钮被点击时触发的信号与我们定义的函数(方法)进行绑定 # 注意:这里没有(),即写函数的名字,而不是名字(),目前是初始化,只有接受信号的时候才调用 
btn.clicked.connect(self.click_my_btn) def click_my_btn(self, arg): # 
槽函数,点击按钮则调用该函数 # 这里的参数正好是信号发出,传递的参数 print("点击按钮啦~", arg) if __name__ == 
'__main__': app = QApplication(sys.argv) w = MyWindow() w.show() app.exec() 
二.自定义信号
除了接收Qt自带的信号之外,我们也可以自行定义信号,在合适的时机,自行发射信号
自定义信号需要使用到 pyqtSignal来声明信号【类变量】 ,并且需要在类中的函数之外声明
本例子中使用了两组信号和槽,一个是自定义的信号“
my_signal = pyqtSignal(str) 
还有一个是系统自带的
“”clicked“” 
程序运行说明:
点击“开始检测”按钮,触发槽函数“check”,
if i % 5 == 0: # 表示发射信号 对象.信号.发射(参数) self.my_signal.emit(msg + "【发现漏洞】") 
同时触发上述my_signal的信号,紧接着调用“
my_slot 
”函数,运行里面的代码:
import sys import time from PyQt5.QtWidgets import * from PyQt5.QtCore import 
* class MyWindow(QWidget): # 声明一个信号 只能放在函数的外面 my_signal = pyqtSignal(str) def 
__init__(self): super().__init__() self.init_ui() self.msg_history = list() # 
用来存放消息 def init_ui(self): self.resize(500, 200) # 创建一个整体布局器 container = 
QVBoxLayout() # 用来显示检测到漏洞的信息 self.msg = QLabel("") self.msg.resize(440, 15) # 
print(self.msg.frameSize()) self.msg.setWordWrap(True) # 自动换行 
self.msg.setAlignment(Qt.AlignTop) # 靠上 # 
self.msg.setStyleSheet("background-color: yellow; color: black;") # 创建一个滚动对象 
scroll = QScrollArea() scroll.setWidget(self.msg) # 创建垂直布局器,用来添加自动滚动条 v_layout 
= QVBoxLayout() v_layout.addWidget(scroll) # 创建水平布局器 h_layout = QHBoxLayout() 
btn = QPushButton("开始检测", self) # 绑定按钮的点击,点击按钮则开始检测 
btn.clicked.connect(self.check) h_layout.addStretch(1) # 伸缩器 
h_layout.addWidget(btn) h_layout.addStretch(1) # 操作将要显示的控件以及子布局器添加到container 
container.addLayout(v_layout) container.addLayout(h_layout) # 设置布局器 
self.setLayout(container) # 绑定信号和槽 self.my_signal.connect(self.my_slot) def 
my_slot(self, msg): # 更新内容 print(msg) self.msg_history.append(msg) 
self.msg.setText("<br>".join(self.msg_history)) self.msg.resize(440, 
self.msg.frameSize().height() + 15) self.msg.repaint() # 更新内容,如果不更新可能没有显示新内容 
def check(self): for i, ip in enumerate(["192.168.1.%d" % x for x in range(1, 
255)]): msg = "模拟,正在检查 %s 上的漏洞...." % ip # print(msg) if i % 5 == 3: # 表示发射信号 
对象.信号.发射(参数) self.my_signal.emit(msg + "【发现漏洞】") time.sleep(0.01) if __name__ 
== '__main__': app = QApplication(sys.argv) w = MyWindow() w.show() app.exec() 
 
问题,在运行的过程中,如果运行过程中,移动窗口或者其他操作,可能会出现“无反应”现象
可以通过多线程解决此问题