Pyqt5小案例,界面与逻辑分离的小计算器程序

news2025/6/7 21:12:13

直接看下最终效果:

使用技术总结

  • 使用Designer设计界面

  • 使用pyuic5命令导出到python文件

  • 新建逻辑处理文件,继承pyuic5导出的文件的类,在里面编写信号与槽的处理逻辑

使用Designer设计界面

要使用Designer,安装一个Python库即可:

pip install -i https://pypi.tuna.tsinghua.edu.cn/simple pyqt5 pyqt5-tools

我用的是python 3.9版本,发现3.10会报错,3.9没问题

在此过程中,用到了如下组件:

  • QLabel、QlineEdit、QPushButton

  • 先从小范围再到大范围,进行布局管理

  • 按钮的上下方,加上弹簧

然后保存成ui文件

使用pyuic5将ui文件导出成python文件

然后执行如下命令:

pyuic5 -o computer.py computer.ui

即可生成computer.py

为什么要用界面与逻辑分离

主要是用Designer设计的界面,肯定要多次调整的。如果每次调整后,生成新的py文件,就会把自己写的代码给覆盖了。

因此,最好是界面的ui生成的python代码,和自己的分离。

用继承机制就可以:

先看下pyuic5生成的代码文件(如下代码是自动生成的,不用多看):

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

# Form implementation generated from reading ui file 'computer.ui'
#
# Created by: PyQt5 UI code generator 5.15.4
#
# WARNING: Any manual changes made to this file will be lost when pyuic5 is
# run again.  Do not edit this file unless you know what you are doing.


from PyQt5 import QtCore, QtGui, QtWidgets


class Ui_Form(object):
    def setupUi(self, Form):
        Form.setObjectName("Form")
        Form.resize(1006, 577)
        self.verticalLayout = QtWidgets.QVBoxLayout(Form)
        self.verticalLayout.setObjectName("verticalLayout")
        self.horizontalLayout = QtWidgets.QHBoxLayout()
        self.horizontalLayout.setObjectName("horizontalLayout")
        self.label = QtWidgets.QLabel(Form)
        self.label.setObjectName("label")
        self.horizontalLayout.addWidget(self.label)
        self.lineEdit_num01 = QtWidgets.QLineEdit(Form)
        self.lineEdit_num01.setObjectName("lineEdit_num01")
        self.horizontalLayout.addWidget(self.lineEdit_num01)
        self.verticalLayout.addLayout(self.horizontalLayout)
        self.horizontalLayout_2 = QtWidgets.QHBoxLayout()
        self.horizontalLayout_2.setObjectName("horizontalLayout_2")
        self.label_2 = QtWidgets.QLabel(Form)
        self.label_2.setObjectName("label_2")
        self.horizontalLayout_2.addWidget(self.label_2)
        self.lineEdit_num02 = QtWidgets.QLineEdit(Form)
        self.lineEdit_num02.setObjectName("lineEdit_num02")
        self.horizontalLayout_2.addWidget(self.lineEdit_num02)
        self.verticalLayout.addLayout(self.horizontalLayout_2)
        spacerItem = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
        self.verticalLayout.addItem(spacerItem)
        self.horizontalLayout_3 = QtWidgets.QHBoxLayout()
        self.horizontalLayout_3.setObjectName("horizontalLayout_3")
        self.pushButton_add = QtWidgets.QPushButton(Form)
        self.pushButton_add.setObjectName("pushButton_add")
        self.horizontalLayout_3.addWidget(self.pushButton_add)
        self.pushButton_minus = QtWidgets.QPushButton(Form)
        self.pushButton_minus.setObjectName("pushButton_minus")
        self.horizontalLayout_3.addWidget(self.pushButton_minus)
        self.pushButton_multi = QtWidgets.QPushButton(Form)
        self.pushButton_multi.setObjectName("pushButton_multi")
        self.horizontalLayout_3.addWidget(self.pushButton_multi)
        self.pushButton_divide = QtWidgets.QPushButton(Form)
        self.pushButton_divide.setObjectName("pushButton_divide")
        self.horizontalLayout_3.addWidget(self.pushButton_divide)
        self.verticalLayout.addLayout(self.horizontalLayout_3)
        spacerItem1 = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
        self.verticalLayout.addItem(spacerItem1)
        self.label_result = QtWidgets.QLabel(Form)
        font = QtGui.QFont()
        font.setPointSize(18)
        self.label_result.setFont(font)
        self.label_result.setStyleSheet("QLabel{\n"
"color:red;\n"
"}")
        self.label_result.setText("")
        self.label_result.setObjectName("label_result")
        self.verticalLayout.addWidget(self.label_result)

        self.retranslateUi(Form)
        QtCore.QMetaObject.connectSlotsByName(Form)

    def retranslateUi(self, Form):
        _translate = QtCore.QCoreApplication.translate
        Form.setWindowTitle(_translate("Form", "Form"))
        self.label.setText(_translate("Form", "请输入第一个数字:"))
        self.label_2.setText(_translate("Form", "请输入第二个数字:"))
        self.pushButton_add.setText(_translate("Form", "+"))
        self.pushButton_minus.setText(_translate("Form", "-"))
        self.pushButton_multi.setText(_translate("Form", "*"))
        self.pushButton_divide.setText(_translate("Form", "/"))

在里面有个类,叫做:Ui_Form

我们可以继承这个类,然后编写编写自己的逻辑即可

import sys

from computer import Ui_Form
from PyQt5.QtWidgets import QApplication, QWidget


class MyUiComputer(Ui_Form):
    def __init__(self, window):
        super().__init__()
        self.setupUi(window)
        self.pushButton_add.clicked.connect(self.do_compute("+"))
        self.pushButton_minus.clicked.connect(self.do_compute("-"))
        self.pushButton_multi.clicked.connect(self.do_compute("*"))
        self.pushButton_divide.clicked.connect(self.do_compute("/"))

    def do_compute(self, method):
        def func():
            try:
                num01 = self.lineEdit_num01.text()
                num02 = self.lineEdit_num02.text()
                if method == "+":
                    self.label_result.setText("计算结果:" + str(int(num01) + int(num02)))
                elif method == "-":
                    self.label_result.setText("计算结果:" + str(int(num01) - int(num02)))
                elif method == "*":
                    self.label_result.setText("计算结果:" + str(int(num01) * int(num02)))
                elif method == "/":
                    self.label_result.setText("计算结果:" + str(int(num01) / int(num02)))
            except Exception as e:
                self.label_result.setText(f"error: {e}")

        return func


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

    window = QWidget()

    my_ui_computer = MyUiComputer(window)
    window.show()

    sys.exit(app.exec_())

这是我们自己的代码,有几点:

1、需要自己初始化 QWidget 作为入口窗口 然后调用pyuic5生成代码中的 self.setupUi(window) 设置主窗口;

2、这个代码主要是编写自己的信号与槽的链接

3、这个代码也有一个知识,connect的槽函数,是可以传参数的,技巧就是自己的函数返回一个函数 见do_compute函数

总结

依然回顾下pyqt5开发的流程:

  • 用Designer设计界面

  • 用pyuic5将ui文件导出成python文件

  • 用继承的方法,编写自己的类文件,实现自己的信号处理逻辑

  • 自己的代码中,初始化入口的Window,传给pyuic5的文件作为启动入口

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/334597.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Qml学习——鼠标事件处理MouseArea

最近在学习Qml,但对Qml的各种用法都不太熟悉,总是会搞忘,所以写几篇文章对学习过程中的遇到的东西做一个记录。 学习参考视频:https://www.bilibili.com/video/BV1Ay4y1W7xd?p1&vd_source0b527ff208c63f0b1150450fd7023fd8 其…

解决:FTP协议路径在资源管理器打开,总是默认跳转到某个浏览器打开

解决:FTP协议路径在资源管理器打开,总是默认跳转到某个浏览器打开一问题描述:(1)电脑原本能够在资源管理器里面,正确打开ftp协议路径地址,不会自动跳转到其他浏览器里面。(2&#xf…

【自学Docker 】Docker search命令

大纲 Docker search命令 docker search命令教程 docker search 命令用于从 Docker Hub 查找镜像。 docker search命令语法 haicoder(www.haicoder.net)# docker search [OPTIONS] TERMdocker search命令参数 参数描述docker search --filter设置过滤条件。docker search -…

Set集合的特点,HashSet去重的几个重要问题

Set集合的特点:无下标,无序(新增顺序和遍历顺序不一致,新增顺序不影响遍历顺序,而且有一个固定顺序),去重(不允许重复记录)public class TestOne {public static void main(String[] args) {// Set集合的特点&#xff…

Python 3 中文编码 ,首先要会

Python 中文编码 前面章节中我们已经学会了如何用 Python 输出 “Hello, World!”, 英文没有问题, 但是如果你输出中文字符 “你好,世界” 就有可能会碰到中文编码问题。 Python 文件中如果未指定编码,在执行过程会出现报错&…

Android Camera(三)Camera Service的初始化代码分析(AndroidR)

highlight: a11y-dark 1.APP CameraManager初始化 CameraManager初始化代码较少,放在这里进行讲解,CameraManager是一个用来管理相机的系统服务,可以通过此服务获取到相机设备和相机信息,在frameworks/base/core/java/android/a…

下载好python后,怎么打开,python怎么进入编程界面

大家好,给大家分享一下python下载安装好后怎么打开,很多人还不知道这一点。下面详细解释一下。现在让我们来看看! 1、python第一次怎么打开 python第一次打开方法如下: winr快捷键打开运行窗口,在运行窗口输入栏中输入…

基于C#制作一个飞机大战小游戏

此文主要基于C#制作一个飞机大战游戏,重温经典的同时亦可学习。 实现流程1、创建项目2、界面绘制3、我方飞机4、敌方飞机5、子弹及碰撞检测实现流程 1、创建项目 打开Visual Studio,右侧选择创建新项目。 搜索框输入winform,选择windows窗体…

线程的生命周期和状态中方法的详解

点个关注,必回关 一、线程的生命周期和状态图: 图一: 图二 二、线程状态 1:新建 2:就绪(拿到执行权开始运行) 3:运行 4:阻塞 5:死亡 三、线程生命周期…

CUDA中的动态并行

CUDA的动态并行 文章目录CUDA的动态并行1. 介绍1.1. 简述警告:与父块的子内核显式同步(即在设备代码中使用 cudaDeviceSynchronize())在 CUDA 11.6 中已弃用,并计划在未来的 CUDA 版本中删除。1.2. 术语2. 执行环境和内存模型2.1.…

ASP.NET CORE 3.1 MVC“指定的网络名不再可用\企图在不存在的网络连接上进行操作”的问题解决过程

ASP.NET CORE 3.1 MVC“指定的网络名不再可用\企图在不存在的网络连接上进行操作”的问题解决过程 我家里的MAC没这个问题。这个是在windows上发生的。 起因很简单我用ASP.NET CORE 3.1 MVC做个项目做登录将数据从VIEW post到Controller上结果意外的报了错误。 各种百度都说…

不要再问Spring是如何解决循环依赖了

1、什么是循环依赖? 循环依赖主要来次三个方面,第一种A相互依赖,第二种是 A依赖B,B依赖A,第三种是A依赖B,B依赖C,C依赖A。 总结一句话就是对象之间形成环形依赖。 代码如下: 第一…

FPGA时序约束与分析 (1)--- 约束工具及相关命令的使用

注意: 时序约束辅助工具或者相关的TCL命令,都必须在 open synthesis design / open implemention design 后才能有效运行。 1、时序约束辅助工具 2、查看相关时序信息 3、一般的时序约束顺序 1、 时序约束辅助工具(1)时序约束编辑…

蓝桥杯刷题练习Python(笔记)

# a1 A # a2 A B A # a3 ABA C ABA # a4 ABACABADABACABAnum int(input(N < 26:)) # 26 个字母 str "ABCDEFGHIJKLMNOPQRSTUVWXYZ"if num 1:a A # str[0] else:a Afor i in range(1, num, 1): # b str[i]c a b aa c# print(str[i])# print(a) …

【React】从 0 开始学 React —— 实现井字棋小游戏

目录1 React 简介2 实现井字棋小游戏2.1 初始化2.2 props2.3 setState2.4 状态提升2.5 副本2.6 简化组件2.7 key2.8 小结3 核心概念1 React 简介 React 是一个声明式&#xff0c;高效且灵活的用于构建用户界面的 JavaScript 库 官网提供了两种学习思路 如果你喜欢边做边学&am…

Vue3 组件之间的通信

组件之间的通信 经过前面几章的阅读&#xff0c;相信开发者已经可以搭建一个基础的 Vue 3 项目了&#xff01; 但实际业务开发过程中&#xff0c;还会遇到一些组件之间的通信问题&#xff0c;父子组件通信、兄弟组件通信、爷孙组件通信&#xff0c;还有一些全局通信的场景。 …

【GPLT 二阶题目集】L2-043 龙龙送外卖

参考地址&#xff1a;AcWing 4474. 龙龙送外卖&#xff08;杂题选讲&#xff09; 作者&#xff1a;yxc 感谢y总&#xff01; 龙龙是“饱了呀”外卖软件的注册骑手&#xff0c;负责送帕特小区的外卖。帕特小区的构造非常特别&#xff0c;都是双向道路且没有构成环 —— 你可以…

Spring Cloud Alibaba Sentinel 控制台

简介 Sentinel 控制台是流量控制、熔断降级规则统一配置和管理的入口&#xff0c;它为用户提供了机器自发现、簇点链路自发现、监控、规则配置等功能。在 Sentinel 控制台上&#xff0c;我们可以配置规则并实时查看流量控制效果。 下载 有以下两种方式获取Sentinel控制台 下…

进阶技术:Linux Arm32是如何调用C Main的

前言&#xff1a; Linux x64通过qemu的rdinit方式调用的C Main&#xff0c;实际上是通过load_elf_binary(加载和解析elf)和start_thread(设置Ip和sp)&#xff0c;用缺页异常来调用。关于这点可以看这篇文章&#xff1a;点击查看。那么Arm32里面是如何调用C Main的呢&#xff1f…

Android 进阶——Framework 核心之Binder 相关预备理论(一)

文章大纲引言一、进程的内存空间和进程隔离二、Linux 系统内存的用户空间和内核空间1、用户空间&#xff08;User Space&#xff09;2、内核空间&#xff08;Kernel Space&#xff09;三、Linux IPC 原理1、内核态和用户态2、IPC 步骤四、内核模块和驱动五、Binder1、Binder IP…