Pyqt5 - 如何用温度更新标签(ds18b20)

时间:2018-02-27 19:04:58

标签: python-3.x pyqt5 raspberry-pi3 temperature qlabel

我试图在附有ds18b20传感器的树莓派3上使用pyqt5构建温度监测器。我是python和pyqt5的新手。

Gui文件(frameless.py)是用QT 4设计师开发的。 Gui应该作为更环境的传感器(例如pH EC等)的基本实例。实际代码(runframeless.py)如下所示。

我很清楚,如果要同时读取更多的Senosor数据,最好使用线程(或多线程),但目前我遇到了一个非常简单的问题。

问题: 温度显示在textlabel中,但def read_temp(self)中的温度值未更新。

问题: 任何人都可以解释为什么它没有更新并帮助我获得正确的代码,以便当温度真正发生变化时,gui中显示的温度值会发生变化吗?

runframeless.py

# -*- coding: utf-8 -*-
# file: runframeless.py)
#-create a skeleton class(es) for Raspberry Pi GUI-

# need this
import sys
import time
import datetime
import os
import glob
#----This gets the Qt stuff------------------------
from PyQt5 import QtGui
from PyQt5 import QtCore
from PyQt5 import QtWidgets

from PyQt5.QtCore import Qt

from PyQt5.QtWidgets import QMainWindow
from PyQt5.QtWidgets import QApplication
from PyQt5.QtWidgets import QLabel


# Import QtCreator/qtdesigner file
import frameless

os.system('modprobe w1-gpio')
os.system('modprobe w1-therm')

base_dir = '/sys/bus/w1/devices/'
device_folder = glob.glob(base_dir + '10*')[0]
device_file = device_folder + '/w1_slave'

#-------------------------------------------------
#----class(es) for our Raspberry Pi GUI-----------
#-------------------------------------------------

class MainWindow(QMainWindow, frameless.Ui_MainWindow): 
    def __init__(self):
        super(self.__class__, self).__init__()
        self.setupUi(self) # gets defined in the UI file

        self.label.setText(str(self.read_temp()))


        self.timer = QtCore.QTimer(self)
        self.timer.setInterval(1000)
        self.timer.timeout.connect(self.read_temp)
        self.timer.start()    

    def read_temp(self):
        f = open(device_file, 'r')
        lines = f.readlines()
        f.close()
        time.sleep(.1)
        while lines[0].strip()[-3:] != 'YES':
            time.sleep(.1)
            #lines = self.read_temp()
        equals_pos = lines[1].find('t=')
        if equals_pos != -1:
            temp_string = lines[1][equals_pos+2:]
            temp_c = float(temp_string) / 1000.0
            return temp_c


if __name__ == "__main__": 
    app = QApplication(sys.argv)
    window = MainWindow()
    window.show()

    sys.exit(app.exec_())

file:frameless.py

# -*- coding: utf-8 -*-

# Form implementation generated from reading ui file 'frameless.ui'
#
# Created: Tue Feb 27 17:31:49 2018
#      by: PyQt5 UI code generator 5.3.2
#
# WARNING! All changes made in this file will be lost!

from PyQt5 import QtCore, QtGui, QtWidgets
from PyQt5.QtCore import Qt

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        self.setWindowFlags(Qt.FramelessWindowHint) # frameless
        MainWindow.resize(800, 480)
        MainWindow.setStyleSheet("background:rgb(0, 0, 0);")
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setStyleSheet("")
        self.centralwidget.setObjectName("centralwidget")
        self.pushButton = QtWidgets.QPushButton(self.centralwidget)
        self.pushButton.setGeometry(QtCore.QRect(660, 430, 101, 31))
        self.pushButton.setStyleSheet("background: rgb(255, 255, 255) ")
        self.pushButton.setObjectName("pushButton")
        self.widget = QtWidgets.QWidget(self.centralwidget)
        self.widget.setGeometry(QtCore.QRect(10, 170, 751, 41))
        self.widget.setObjectName("widget")
        self.horizontalLayout = QtWidgets.QHBoxLayout(self.widget)
        self.horizontalLayout.setContentsMargins(0, 0, 0, 0)
        self.horizontalLayout.setObjectName("horizontalLayout")
        self.label_2 = QtWidgets.QLabel(self.widget)
        self.label_2.setStyleSheet("background: rgb(255, 255, 255) ")
        self.label_2.setObjectName("label_2")
        self.horizontalLayout.addWidget(self.label_2)
        self.label = QtWidgets.QLabel(self.widget)
        self.label.setStyleSheet("background: rgb(255, 255, 255) ")
        self.label.setObjectName("label")
        self.horizontalLayout.addWidget(self.label)
        MainWindow.setCentralWidget(self.centralwidget)

        self.retranslateUi(MainWindow)
        self.pushButton.clicked.connect(MainWindow.close)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))
        self.pushButton.setText(_translate("MainWindow", "exit"))
        self.label_2.setText(_translate("MainWindow", "Ds18B20-sensor"))
        self.label.setText(_translate("MainWindow", "TextLabel"))


if __name__ == "__main__":
    import sys
    app = QtWidgets.QApplication(sys.argv)
    MainWindow = QtWidgets.QMainWindow()
    ui = Ui_MainWindow()
    ui.setupUi(MainWindow)
    MainWindow.show()
    sys.exit(app.exec_())

1 个答案:

答案 0 :(得分:0)

在调试代码时,逻辑上逐步完成程序的工作是值得的。

首先,此行self.label.setText(str(self.read_temp()))__init__方法中运行一次。这会调用返回温度的self.read_temp()方法并将其放在文本框中。

接下来,您在QTimer方法中创建__init__,每秒调用self.read_temp()。因此,当计时器每秒触发一次,self.read_temp运行时,温度被读出,并返回到QTimer用于调用方法的任何内部代码。

此时QTimer会抛弃返回值,因为它不想要它,不能使用它等等。

正如您现在可能看到的那样,它没有更新的原因是因为您从未告诉它更新。

我建议在self.label.setText(str(temp_c))行之前添加行return temp_c,以便从每秒运行的代码更新标签。