QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2207|回复: 0
打印 上一主题 下一主题

超全!Python图形界面框架PyQt5使用指南!

[复制链接]
字体大小: 正常 放大
杨利霞        

5273

主题

82

听众

17万

积分

  • TA的每日心情
    开心
    2021-8-11 17:59
  • 签到天数: 17 天

    [LV.4]偶尔看看III

    网络挑战赛参赛者

    网络挑战赛参赛者

    自我介绍
    本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。

    群组2018美赛大象算法课程

    群组2018美赛护航培训课程

    群组2019年 数学中国站长建

    群组2019年数据分析师课程

    群组2018年大象老师国赛优

    跳转到指定楼层
    1#
    发表于 2022-9-15 12:26 |只看该作者 |倒序浏览
    |招呼Ta 关注Ta
    超全!Python图形界面框架PyQt5使用指南!! o6 g% L! A' `' H3 [* L9 P- i

    - l  n) R: w1 \" Q" t- U使用Python开发图形界面的软件其实并不多,相对于GUI界面,可能Web方式的应用更受人欢迎。但对于像我一样对其他编程语言比如C#或WPF并不熟悉的人来说,未必不是一个好的工具。2 ~" N1 I) z* U4 f7 O7 w0 g* _. T9 ?

    . h7 m- C+ }0 A( R常见GUI框架2 ~/ z/ ~: _, i. ^  q' K* n5 N

    0 Y# M1 `+ S! c4 U) T! J0 w2 hPyQt5:Qt是一个跨平台的 C++图形用户界面库。QT一度被诺基亚拥,后出售给芬兰的软件公司Digia Oyj。PyQt5是基于Digia公司Qt5的Python接口,由一组Python模块构成。PyQt5本身拥有超过620个类和6000函数及方法。在可以运行于多个平台,包括:Unix, Windows, and Mac OS。' y. {# L1 \, G' j4 N

    . {9 K, f, J2 J& i. A4 y( \: F) }Pyside6:Pyside是QT公司官方提供的Python包,上一版本为Pyside2,对应的是QT5,最新版命名规则进行了调整,更改为Pyside6,对应的是QT6版本。由于官方出品的比较看好,缺点是发布比较晚,网上的资料没有PyQt5多。
    * R, \1 a. u; O' J, Z$ B; O; r, d3 [$ y  w) @6 v
    Tkinter:Python内置的GUI框架,使用TCL实现,Python中内嵌了TCL解释器,使用它的时候不用安装额外的扩展包,直接import,跨平台。不足之处在于UI布局全靠代码实现,只有15种常用部件,显示效果简陋。
    / j* N# f; ~' H
    ; f4 P7 r7 C" B, f" M9 NPySimpleGUI:PySimpleGUI 是 Tkinter 一层包装。使用 PySimpleGUI 实现自定义 GUI 所需的代码量要比使用 Tkinter 直接编写相同的 GUI 要少得多。! Y# f) ^( K# A) A4 f2 ?
    ' t& l: d: n# t+ v2 o  ]% {0 g" \
    WxPython:wxPython是Python语言对流行的wxWidgets跨平台GUI工具库的绑定。用得比较广泛,跨平台,C++编写,文档少,用户可能就需要根据编程内容对不同平台中的GUI代码做一些调整。遇到问题不好解决,代码布局控件,不直观。, [. ]3 C; l8 X2 f  u. `- n* ?
    6 u) x8 b. U( k( ~4 H% X: J
    Wax:基于wxPython ,为克服wxPython的问题而制作的一个包。
    ( S, L: n6 R' g- W$ F- I+ K  a6 }2 h* z$ h; m  `
    Kivy:主要针对多点触控程序,智能手机平板等,也可以在没有触屏功能的系统上,全平台支持(Windows, Linux, Mac OS X, Android and iOS.)使用Python和cython编写,中文支持差,需要自己下载中文库并且制定路径。# \( g/ }' N+ i7 H
    ( x+ e; g- n+ S# J/ p
    BeeWare:Write once. Deploy everywhere.需要与Kivy配合使用。
    8 x* `% b5 ]8 g& J
    : e1 v) w4 f% u3 cToga:一个使用Python开发原生APP的GUI工具包。Toga由一个具有共享接口的基础组件库组成,以简化与平台无关的GUI开发。Toga适用于Mac OS、Windows、Linux(GTK)以及Android和iOS等移动平台。. ?' v4 K$ E, \  k- F- r% I

    8 ^/ Z5 Z* a7 U8 XEel:一个轻量的 Python 库,用于制作简单的类似于 Electron(但是比它更轻量) 的离线 HTML/JS GUI 应用程序,并具有对 Python 功能(capabilities)和库的完全访问权限。
      E* r5 ^# G2 \7 ^0 r3 o. S$ A  U$ h6 I4 s# Q6 d9 b. q
    Flexx:一个纯 Python 工具包,用来创建图形化界面应用程序。其使用 Web 技术进行界面的渲染。你可以用 Flexx 来创建桌面应用,同时也可以导出一个应用到独立的 HTML 文档。因为使用纯 Python 开发,所以 Flexx 是跨平台的。只需要有 Python 和浏览器就可以运行。. j, ], \" ^5 F/ Y. P2 ~- l

    # s: e- M4 v; F/ y+ i4 Vpywebview是围绕 webview 组件的轻量型跨平台包装器(wrapper),它允许在其自己的本机 GUI 窗口中显示 HTML 内容。它使您可以在桌面应用程序中使用 Web 技术,同时尽最大可能隐藏使用浏览器构建GUI的事实。
    " u+ q- F7 G- h* d& U( J( j% w$ \+ ^" a) `$ f5 Z' i
    enaml:一种能够让你用最小的努力就可以实现高质量GUI界面的的Python框架,也是一种独特的编程语言。enaml将声明性语言与基于约束的布局系统结合在一起,使用户可以轻松地定义灵活布局的UI。enaml应用程序可以在任何支持Python和Qt的平台上运行。6 C' E2 l4 J& \% t' {  Q
    / E5 z, G# T6 o9 |) t% @
    个人想法:太多学不完,先学PyQt5,原因是资料多,学有余力再学pyside6,最后看下PySimpleGUI,看能否解决一些简单问题。4 I6 j: ?5 J- n" x- ^

    1 P1 U1 q: n8 A! TPyQt5简介
    ! j  E! p5 g0 A& n. l
    0 {3 z# ]5 ]  I4 \7 z  N. TPyQt是Qt框架的Python语言实现,由Riverbank Computing开发,是最强大的GUI库之一。PyQt提供了一个设计良好的窗口控件集合,每一个PyQt控件都对应一个Qt控件,因此PyQt的API接口与Qt的API接口很接近,但PyQt不再使用QMake系统和Q_OBJECT宏。
    * z( x, w$ H5 q! C& o: P$ K$ ~" I% R" ?. ^: `7 }
    PyQt5提供GPL版和商业版证书,自由开发者可以使用免费的GPL许可,如果需要将PyQt用于商业应用,则必须购买商业许可。4 }. `- ?5 i* K& c% _

    8 u% C3 n: a' O4 g2 vPyQt5特性如下:4 Y1 H  s2 w; ^$ z2 D
    5 J6 ^9 I: g& L: t4 v+ [6 ?
    基于高性能的Qt的GUI控件集。( L  Z; E9 L: F# D- \0 ?
      X+ F) X1 z; ]: E6 t
    能够跨平台运行在Linux、Window和Mac OS系统上。# W: M6 |6 s' b% h5 x
    , L1 {/ J4 q0 N& g; N: D5 A
    使用信号槽机制进行通信。
    ! o4 F8 V3 c" S( n+ I7 X! y/ A% R0 A
    对Qt库进行完全封装。
    2 N! x3 B4 g8 g$ \- A/ @0 P2 N1 h7 d/ G2 r* G+ F
    可以使用成熟的IDE进行界面设计,并自动生成可执行的Python代码。
    6 O) ]5 K. d" x; Q2 e2 H2 U' Y$ Q
    6 S6 w/ H$ z) o  _提供一整套种类齐全的窗口控件。1 J$ D: r* u$ [
    7 L% ]) F( [1 C) w
    PyQt5是由一系列Python模块组成,有超过620个类,6000个函数和方法,主要模块如下:
    5 n6 J% S7 Y- v* y9 b; z* c% ?/ ^9 a7 H  q" j% y* d
    QtCore:包含了核心的非 GUI 的功能。主要和时间、文件与文件夹、各种数据、流、URLs、mime 类文件、进程与线程一起使用。- R5 y$ v$ N) ]/ c5 @% Q0 B

    4 U. H# e3 H8 q6 j) JQtGui:包含了窗口系统、事件处理、2D 图像、基本绘画、字体和文字类。
    4 S0 o8 }2 Z! |; {4 Z4 o2 v( y1 o
    # m8 F0 k7 n$ [4 @. B- @QtWidgets:包含了一系列创建桌面应用的 UI 元素。2 @/ W( c6 @/ k0 A9 b( v$ [( r

    + d9 p# g: C" Q7 nQtMultimedia:包含了处理多媒体的内容和调用摄像头 API 的类。: X7 a$ {; O+ B( L/ }* y
    - J9 Z2 f) G9 Z6 ?
    QtBluetooth:包含了查找和连接蓝牙的类。
    * ~7 G+ J- o2 u2 ^( O
    / {! A5 [7 G* |4 w, w0 n0 |QtNetwork:包含了网络编程的类,这些工具能让 TCP/IP 和 UDP 开发变得更加方便和可靠。
    8 H& G& v) u9 E5 c: `$ o. D# j( o( j) F4 g  l
    QtPositioning:包含了定位的类,可以使用卫星、WiFi 甚至文本。/ u2 E( w) ~* C# @- N: l

    5 O2 d% U  B6 I% V; t( {Enginio:包含了通过客户端进入和管理 Qt Cloud 的类。5 B2 X9 K" S, P# B, Q; h- j. d# q" _
    7 X: E9 A; _( J
    QtWebSockets:包含了 WebSocket 协议的类。
    $ f9 A# B) ?  x" K/ q% H9 k
    , g2 X" z+ j! S5 W7 J  }  U5 iQtWebKit:包含了一个基 WebKit2 的 web 浏览器。
    : O. G, Y  W  L$ K4 F7 R$ d
    ! r  g: T' k# g/ f3 TQtWebKitWidgets:包含了基于 QtWidgets 的 WebKit1 的类。% u% W' X9 E% _

    , D5 _2 d; ~+ iQtXml:包含了处理 xml 的类,提供了 SAX 和 DOM API 的工具。
    ) _* m" a( n5 M8 L
    7 P% S) Z) m3 j& HQtSvg:提供了显示 SVG 内容的类,Scalable Vector Graphics (SVG) 是一种是一种基于可扩展标记语言 (XML),用于描述二维矢量图形的图形格式(这句话来自于维基百科)。
    " `* h+ k! N" C$ d( x5 }  }4 d5 S& \9 ^) }/ h8 h# o! N
    QtSql:提供了处理数据库的工具。
    ) E0 N! A% q: E
    + g  r" D7 [/ m! GQtTest:提供了测试 PyQt5 应用的工具。
    9 ?) l- N4 E& O- ?# ~6 k+ B1 h6 ~& t
    PyQt5的安装  L( B$ J9 a4 j/ L5 k/ L0 Y
    , }3 d: J+ f0 A0 c( ~4 z% J1 e
    由于后期要使用fbs进行打包,fbs对Python 3.7以后的版本可能存在兼容问题,所以我选择了Python 3.6.8进行了整个环境的搭建。主要内容为:Python + PyCharm + PyQt5
    % C  r" P0 g2 d+ ~( P0 m) X9 u% X, O
    安装PyQt5
    9 W* U8 d# X% d9 E% q0 |
    ! V, s( o0 G# Npip install pyqt5
    & t( q4 _5 s8 @" L% l( u, F- F, i
    * A& m4 i/ j) a  Ipip install pyqt5-tools+ H  ?& C6 J8 |8 Q9 c
    其中pyqt5-tools为Qt Designer拖拽式的界面设计工具。安装过程中可能会报如下错误:
      e6 S- m! |3 ^0 \6 p$ i) _
    * F4 V, M: B% b1 a( j9 s/ h9 eqt5-tools 5.15.2.1.2 has requirement click~=7.0, but you'll have click 8.0.1 which is incompatible.+ M6 |2 ]0 c) P$ v
    解决方案:
    6 ~$ I& k) r) o* ^" K! Y
    5 k# Q4 Z8 I/ W  V! X4 T( Z+ kpip install click~=7.0
    6 |6 R: O9 Q! d  RQt Designer的配置. t. M5 p8 d* n; H  S8 O" v' E+ y
    4 n. O4 I' x+ B9 \- {
    Qt Designer 是通过拖拽的方式放置控件,并实时查看控件效果进行快速UI设计。
    , ?. [5 [4 m* ~2 {# M* z# ^& E' Z/ A0 V2 R: M/ @

    & O& D/ V: J0 U3 p7 z% G+ i' `0 J  z* q$ S1 v! N5 z
    整个画面的构成:, G5 X. I' q' ^7 ^8 G. {
    $ w7 `3 g- `6 b* X/ ~: A: a; t
    左侧的“Widget Box”就是各种可以自由拖动的组件% a, `8 h( M. K0 i/ p

      N; K$ r( \+ ^  r. T中间的“MainWindow – untitled”窗体就是画布
    6 j; j: G9 v  N; I! H
    $ h" B, s) W, M8 e% _, O8 h右上方的”Object Inspector”可以查看当前ui的结构
    ; C/ k/ U" r5 t% i" e9 |; X. J! `* B( x0 W- t. T/ j% r* J
    右侧中部的”Property Editor”可以设置当前选中组件的属性
    2 a8 l+ b! J! C" g3 g% @. q
    ; \+ l5 h* D$ r% N. K右下方的”Resource Browser”可以添加各种素材,比如图片,背景等等+ O7 A5 E1 U6 d) o/ J" U/ C( k
    - e0 l* T) A$ g$ X# t  W
    最终生成.ui文件(实质上是XML格式的文件),可直接使用,也可以通过pyuic5工具转换成.py文件。' [4 y4 Z8 g. z
    - `& V$ g, i' W' y
    QtDisigner配置
    ! \( Q( n: K- M- ~: f$ w$ {' K% O
    / v2 c9 g2 |3 a5 W7 f& W1 d0 p' t在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:
    2 h9 `% ], ~2 f9 ^+ T- g
    ( A: d$ {  k! w& T/ O0 R, vName: QtDisigner% o- p9 |* p$ R& v
    3 I! l: r6 b; }. P1 v( I: H3 ?* H
    Program : D:\Program Files\Python36\Lib\site-packages\qt5_applications\Qt\bin\designer.exe # 请根据实际修改
    9 s5 q! O; j* D0 l4 l% f* d5 J5 P+ Y9 M7 g7 @3 f+ f
    Working directory: $FileDir$/ y# N4 F6 S1 W$ ]7 h" r
    PyUIC配置
      h1 R0 N. U: j5 u: l0 Q7 m* X
    * W" ]  n8 @3 J: ?5 s# ]5 }8 N/ MPyUIC主要是把Qt Designer生成的.ui文件换成.py文件。
    2 J$ S& [& O$ N2 h, _8 N; f; x1 O/ r8 J1 F$ z7 x7 @8 F! B
    在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:4 ?/ f* ^3 I  p4 [0 w

    + R* F. Y  B- mName: PyUIC+ h' i. T2 R3 |$ d7 t

    ! U* U2 B% s6 G$ }1 D9 jProgram : D:\Program Files\Python36\python.exe # 当前Python目录,请根据实际修改! h8 }4 h1 B# M

    9 _7 Z# a9 A# n1 XArguments: -m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py
    6 s% w( {, U% q3 N; e
    ( Z. K0 P+ X* s( e; ~Working directory: $FileDir$2 W6 a+ ~) y( ?2 g" R9 ]
    PyRCC配置) N: j  ]3 h: D, d0 C( l# @2 D

    2 e  w' X- c& Z' c: e% sPyRCC主要是把编写的.qrc资源文件换成.py文件。在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:/ [9 Y% C+ P/ X) W

    ! c1 G; w+ j- b5 q' |! P& @Name: PyRCC3 A: `/ ]1 z! f5 R" O- ^

    8 F9 I* f) ~9 _0 X% hProgram: D:\Program Files\Python36\pyrcc5.exe # 当前rcc工具目录,请根据实际修改
    # i1 S2 U  Q( X5 U' u
    ( r3 Z0 }6 m2 j7 j- e% }3 @$ v: Y5 PArguments: $FileName$ -o $FileNameWithoutExtension$_rc.py; `" a5 D/ M7 b' R0 b, Q# O

    * q5 J+ x4 @) b4 l: CWorking directory: $FileDir$
    + ]7 b% o' }5 @1 PPyQt5使用示例
    6 k. h: r% y/ W  b. \9 H5 u3 ?2 _2 Y& {$ |. G; F
    创建一个空白的界面:
    3 v0 p% t0 q: E# W
      `5 {$ \( `2 H; E+ u; d3 Vimport sys
    $ k' ^' H: o9 R/ \; O1 S
    ) `1 h8 H1 r0 t$ Z) Y# M, {9 d0 c! Ufrom PyQt5.QtWidgets import QApplication, QMainWindow, QLabel1 x; M5 [/ l+ P
    5 w% {/ D9 n- Y
    app = QApplication(sys.argv)5 Z2 |# ]; s/ r
    6 Z- q6 L$ Z0 k$ o6 K
    win = QMainWindow()6 b4 \, [0 t0 N2 v/ J

    8 d  v# a+ G7 T; K6 f" Mwin.setGeometry(400, 400, 400, 300)1 ~  a: S9 O( R/ O1 W' S
    / Q3 t/ k; h2 [3 ^, y" s$ i' I
    win.setWindowTitle("Pyqt5 Tutorial")- K- u* V" t$ O  Q
    7 d- G% d. A1 n' ~+ G6 S; L) O
    win.show(), R8 n. z1 D5 v5 V4 W4 m' u

    + ^/ C. W  I( c* O3 T+ asys.exit(app.exec_())
    0 M6 t1 R8 a: e  L' W" k" J2 E$ e
    : e% B- ^4 k- t& `! o  J2 B
    5 X$ Q/ `* s' q  B; `
    其中:5 o1 }! v0 w' i' V7 @+ `0 \" Q. L' N
    ( c$ z4 u6 A8 \; P: K4 m# r
    Qapplication():每个GUI都必须包含一个Qapplication,argv表示获取命令行参数,如果不用获取,则可以使用[]代替。# Z$ a# j* ]3 n4 U  a3 z. u' r: G1 [
    1 E. z- ^' E# t
    QMainWindow():类似一个容器(窗口)用来包含按钮、文本、输入框等widgets。arg标识可以获取命令行执行时的参数。7 i5 q* ~' I1 c, S

    # \9 \- @/ `6 s5 m) |2 A7 RSetGeometry是用来定义 QMainWindow() 窗口的尺寸, 语法:setGeometry(x, y, width, height ),其中x,y为屏幕上的坐标点。0 y# g: Y6 K) _
    % ]* P- j# f% L4 m- \/ H
    show():用来显示窗口
    ' C  u, g+ g" A5 c5 E# J- ~. q- R4 c! M( \2 B) p7 V: U
    exit(app.exec_()):设置窗口一直运行指导使用关闭按钮进行关闭0 q& M7 x" I! X5 n

    8 \5 D$ q) W# Y4 R+ c% U% mPyQt5支持的常见Widgets有:
    ( s, `3 e  x) ~. O; F  v7 t' e, ^( o; L) |" F" b9 k- [, Z! ^/ W  a
    + i# w/ J8 x# E& g4 C3 T0 ~* J

    - p. A- N9 s$ `; X% K$ M9 ~从上到下,从左到右依次为:Qlabel、QcomboBox、QcheckBox、QradioButton、QpushButton、QtableWidget、QlineEdit、Qslider、QProgressBar
    * ~* t' T9 C! n0 Q/ a1 G
    8 P+ n( H9 g2 i6 ]对于使用Pyqt5设置文本内容,我们使用Qlabel:
    # X5 G+ [, x9 i+ j' N1 K3 P8 m7 i7 i. ^, O( ^# s
    import sys4 T0 I5 }& A! c

    5 \* U5 M: K/ j8 Ffrom PyQt5.QtWidgets import QApplication, QMainWindow, QLabel
    ) ]0 g4 h4 D7 }7 t, B! l  i/ ~' O) G! k
    app = QApplication(sys.argv)
      K2 b9 D% Y& h  c% }, _1 H: C8 Q' g* {
    win = QMainWindow()
    , @# Y: Y1 ]' j$ q  c7 @
    9 {4 [) L  @6 a: F9 z; I+ d: }win.setGeometry(400, 400, 400, 300)
    6 B* O/ `& \7 d0 C% M0 p' c# q) b% f8 R6 O. d9 N; X1 m
    win.setWindowTitle("Pyqt5 Tutorial")
    - r7 x: ~  O* I, L8 D: o
    5 _4 m9 p+ T) _8 H) e( T  P\# Label Text4 F7 d6 B7 v# J5 A
    " s7 m$ h$ x' r8 \
    label = QLabel(win)
    : Y9 s# Z1 d7 p0 W2 _1 c8 G% ]- ~) Q6 e9 c
    label.resize(200, 100)2 f6 }: ?8 H6 q5 x4 x
    ; P. \1 }( q- v0 k2 @: x  K0 q
    label.setText("Hi this is Pyqt5")
    6 J, E* c4 V; L$ w( H- B- u0 N! D9 q! N2 f6 b! B
    label.move(100, 100); ~" B  b8 E; e1 A1 E% u: m# E

    9 R9 P; _+ F" h3 j. ^. d2 |7 swin.show(). U& d1 O& a$ u0 e
    8 s8 z4 t6 D# `& J/ G2 T
    sys.exit(app.exec_()). J% d9 m" P% G% \( i8 S+ ]3 \

    * t7 K+ L3 q( n; ~4 i4 c' q3 X
    1 d: |% P# {' `& ^( e$ b) f! D  j2 j+ f; U. @! b  V; b" r
    按钮与事件:
      v; H$ ]* B4 k: E+ p( f' L. b! z& x9 ]( `% P* w
    import sys, ^. H- L: x! n) [
    5 _9 Q1 ~) `* n, T; ~5 F
    from PyQt5.QtWidgets import QApplication, QMainWindow, QPushButton
    ' d  r7 F0 q2 e0 [2 i
    0 k/ O) z  X: H' |5 z! Q2 mdef click():
    6 S# ?- G. y, v" W0 }. c+ @$ j8 P1 S% A9 W; K
        print("Hy Button is clicked!")
    " s2 X( Q9 i# J, k) D  t( h" R) X7 p: N' A; D+ X$ U$ `
    app = QApplication(sys.argv)3 B1 A; r8 t: s0 k
    . u" r: I. ^" j8 b* f
    win = QMainWindow()# @, q3 ^& Z$ ~* V1 ~) u) {
    4 r' M# W. n2 ]$ F
    win.setGeometry(400, 400, 400, 300)7 V, w: T* H9 ]

    * k! d0 e* f# ~- g, m2 Z3 ]win.setWindowTitle("Pyqt5 Tutorial")
    - T$ a' j( |& C( R; w
    ' n+ p9 N  ?0 \6 F! v7 L8 O* m1 l\# Button
    8 {& I: b9 y8 L* B# H
    * g& Q7 n. l. Dbutton = QPushButton(win)
    1 X: m* M; r, R/ a
    6 B8 `+ g5 [$ k( @button.resize(200, 100)  v+ @$ {  T$ v8 ~, r9 s4 J- a

    ' K4 z5 ^) x/ z6 S8 j6 S8 g/ ^button.setText("Hi! Click Me")
    ) }, g8 g/ n0 E$ U0 d1 T
    . V! |/ e# D  Z: `button.move(100, 100)& ]6 K! r3 I2 x9 P% P" X( j. L4 w8 a

    0 t/ ?9 B6 h1 r" w) Tbutton.clicked.connect(click)0 r3 p4 W" M9 h1 z, s  N$ r

    * j' c, b# t  R- V, A7 l. Xwin.show()3 q3 R0 @, l$ P+ U# ?4 N" G
    1 E, n$ r+ e' o* e8 b8 V
    sys.exit(app.exec_())
    ) I5 _- y# w; B: G& \, p/ W3 p$ m6 f) M2 v( y
    7 m9 w; F* e9 g% c2 l# a

    $ E" N5 s1 o2 [" R& i4 v) Z; g. mbutton.clicked.connect() 在按钮点击后执行特定的事件。
    " s, o5 P' x* Z1 o7 {5 z! {) @+ W& j! |0 q3 l- d+ E7 m4 V/ Y
    PyQt5实战% D  k( x. K& ~3 a/ l. m

    ) t( k8 O& W+ _8 s" ?8 _, ]  U2 z实战项目:简易的天气查询软件9 h6 t! k$ m6 _2 n
    0 Y& p: d' q: Q' |3 r6 B
    1、使用Qt Designer设计一个界面; r  [- G4 h3 _4 B/ N! K

    # E' X8 A/ n6 Y# p3 O7 K5 a- g, c. S- |; p+ q3 |
    . m/ g- t& z4 K& c$ g- N" h
    用到的控件有Button, GroupBox, Label,ComboBox,TextEdit,同时定义了两个按钮queryBtn及clearBtn,分别用来查询及清空天气数据。我们需要绑定槽函数,方法如下:
    $ u6 U9 E% D' ?! ?6 ~5 a
    " k- F( H' d! W8 o- w在Qt Designer右下角选择 信号/槽编辑器,点击+号新增
    5 X" v) u4 f  U8 O( L( b$ H! Y4 N7 Y! A# B$ I/ }- p
    分别选择queryBtn及clearBtn,选择信号 clicked(), 接收者 Dialog 及槽 accept(),(槽函数这里不知道如何定义,后期在代码里再进行修改)$ m+ @: f) [- ~2 ?* C% `) [
    # z0 r# [; D) ]0 f- \
    以上完成后保存为Weather.ui文件。
    * `+ P( A9 J4 d2 S4 c- J9 s" m1 `5 N. {
    2、转换.ui文件为.py文件9 e# g  J; J* x* R

    9 |7 d. C, U6 n+ }- hPyQt5支持直接使用.ui文件:
    ) @$ k5 Q6 ?6 e7 X) [  @% c" _7 D
    ' o3 q# A4 T) G' K* v; _2 F& kimport sys2 E" n- c! x' K5 L$ Z0 O- n6 |3 e8 _
    ) K( C6 B0 [! g3 q: S
    from PyQt5 import QtWidgets, uic, ]1 P3 F2 A; c0 ^5 p
    ) O  F0 |5 H) ^, K% k
    app = QtWidgets.QApplication(sys.argv)
    9 H2 R% V  u$ @8 W" l3 P) p" ]: W
    window = uic.loadUi("mainwindow.ui")6 |$ ^  e6 `& Z  r
    % t* t  B, Y3 `7 {( m- y) D) x8 q
    window.show()
    ' Z0 I/ h% g' v% y; m  U+ N1 m  ~$ ~2 |! h: F
    app.exec()9 _- r% G0 z0 Z& d) M
    但是为了更好的自定义及修改上面的槽函数,可以使用External Tools – PyUIC,即可生成Weather.py,实际运行命令如下:- f6 l( g2 M9 X5 {+ G* u2 ~& j
    ' }# r) t4 z6 \6 H  q# `, ^
    D:\Program Files\Python36\python.exe -m PyQt5.uic.pyuic Weather.ui -o Weather.py
    - L5 A* p! _1 J& b- s  c$ A5 d其中,我们需要把两个按钮绑定的槽函数:* a! k3 r& |2 L3 Y$ }* J
    * L' K! }* v; u# Y0 d' }. C
    \# self.queryBtn.clicked.connect(Dialog.accept)9 N( }3 p* }7 L( u( a
    / t/ r6 Y: |  n
    \# self.clearBtn.clicked.connect(Dialog.accept), @" j$ b1 @/ t  e+ G# U2 W
    + Z, v( R8 v* h# `4 U
    \# 修改为:
    8 d. R+ {, C, i9 y1 z
    # H" Z7 w: u8 }1 a' p1 |self.queryBtn.clicked.connect(Dialog.queryWeather)  h; \# y6 Y/ B% G* Y. g
    7 i, B# Y' ~8 {+ F5 M
    self.clearBtn.clicked.connect(Dialog.clearText)
    - b& `+ J* ~+ s# x' R( {! [最终的Weather.py内容如下:: j: B/ R* {: R) M+ V
    # \' D) z4 Q+ O
    \# -*- coding: utf-8 -*-
    2 L) k3 r6 H/ g
    # j3 w' e* V# t+ H\# Form implementation generated from reading ui file 'Weather.ui'
    0 Y1 ^' N! Q9 S9 l4 U/ V2 l5 J; v: K4 M/ K# D; J
    \#4 R4 l' Q  k  V4 y, q: Q
    % Z. q  {& p/ K" k% i6 _% L" p6 \
    \# Created by: PyQt5 UI code generator 5.15.4( q8 C7 ]: Y8 ^, O, K, a& x. m
    ; Y: W; o" W1 _3 I
    \#& b' f! u9 z4 c8 H( V7 c
    . t' ?* n0 w" R8 k) _
    \# WARNING: Any manual changes made to this file will be lost when pyuic5 is
    + |& k! Z9 y4 n) K: h7 F5 U! o/ X) o7 m# n: [
    \# run again.  Do not edit this file unless you know what you are doing.) j* }0 a* C) G; ^( v! V8 N8 ~
    8 ~% i, g' D" g0 e8 P
    from PyQt5 import QtCore, QtGui, QtWidgets
    4 y, ]) ?" H. k5 W5 N% g
    " Y/ S2 V! G1 S1 i/ U2 j; P. rclass Ui_Dialog(object):4 d7 K4 J( b/ w( `& \

    9 G0 q6 V+ X  J* [    def setupUi(self, Dialog):
    & i/ y" N, a* E( Z4 G$ P% b5 S
    ( l) A2 Z5 \3 `( z        Dialog.setObjectName("Dialog")
    5 e  ]  z9 a6 c4 E& p3 ]
    # j6 I2 z; b6 @: r        Dialog.resize(600, 600)8 g, c7 _- W9 y8 O) b/ P2 h
    % _8 _  L( b+ C8 \/ J
            self.groupBox = QtWidgets.QGroupBox(Dialog). |' Q' b4 W6 x5 m, I$ j8 ~
    ! i0 w$ Y' z. K$ r* c
            self.groupBox.setGeometry(QtCore.QRect(30, 20, 551, 511))
    " _8 Y# {# j. H  e. F' k' A" }  j. {, f& X) h/ P# Z
            self.groupBox.setObjectName("groupBox")) K/ U" M$ x. s0 Q" a5 |6 `  Q  ~+ j
    . E: f, c& B4 B
            self.label_2 = QtWidgets.QLabel(self.groupBox). @  {1 d+ q3 z- m# G$ K
    0 q' ~, L$ {  d" q2 L
            self.label_2.setGeometry(QtCore.QRect(20, 30, 31, 16))) m1 z) S: K: G* H5 p$ r5 K
    : l* r: C, K4 H5 O; B
            self.label_2.setObjectName("label_2")
    * t% v1 }; }  d# @
    0 }) G: V0 b* K. h) x( Y        self.comboBox = QtWidgets.QComboBox(self.groupBox)
    / ]8 U7 h. n# z7 M9 T8 x: V/ I$ U: s  K' M6 ]
            self.comboBox.setGeometry(QtCore.QRect(70, 30, 87, 22))6 T  A1 f$ |1 L$ z. Z0 y/ y
    # I7 x  }+ h0 h2 b. ?, @
            self.comboBox.setObjectName("comboBox")& A" D% U- x+ x8 m

    " S( L" }; Z$ ]' L        self.comboBox.addItem("")
    : T% D: J/ U8 N' ~$ I8 d0 Z1 R( m
            self.comboBox.addItem("")& c6 J# t/ |. k+ P  X/ g, @3 r
    + U" J- A6 A7 U( _" c! e
            self.comboBox.addItem("")
    & T( e# a! D8 n6 U2 V0 o" l+ b' L' V( Z6 c
            self.textEdit = QtWidgets.QTextEdit(self.groupBox)
    6 F' d' @$ h; i/ E8 D1 O2 L6 ]4 j
    ! M, {6 r6 x8 C( M" t& P; B        self.textEdit.setGeometry(QtCore.QRect(20, 70, 491, 411))
    4 h# e1 @& K4 R
    8 Y" a. B6 S1 R" V        self.textEdit.setObjectName("textEdit")
    + \. r- R& X2 A2 m- D0 E1 B; q% }# i4 R- x- G
            self.queryBtn = QtWidgets.QPushButton(Dialog)
    8 h$ [# E* T" |
    + ?& a# f! }- E        self.queryBtn.setGeometry(QtCore.QRect(490, 560, 93, 28))1 w+ {( q' E9 |: f' w. _
    $ B4 L; x; Z# g$ g  d2 @9 M
            self.queryBtn.setObjectName("queryBtn")5 j& B6 @/ B' M, b7 i! N8 f/ c
    , w, [; ]& _+ ?# F  G' y2 A
            self.clearBtn = QtWidgets.QPushButton(Dialog)& K9 M- v& T* g, @* n: C; i! ?

    ; [& b( Q- _: ?5 S+ f& E/ `) h        self.clearBtn.setGeometry(QtCore.QRect(30, 560, 93, 28))+ s  D. B0 _$ G

    ; g0 }# \% ]/ `( }1 i* ]* d        self.clearBtn.setObjectName("clearBtn")) ^' W) l' s1 [
    2 S% u, h3 V1 ^0 Q& C7 t( k! m
            self.retranslateUi(Dialog)% {1 ]4 j% {+ i* x
    3 c- x7 Q" `/ L1 ^% J' @! U+ a
            self.clearBtn.clicked.connect(Dialog.clearText)
    ; U0 O( k8 @) q7 R
    ' s' T, z, @8 O  _1 K        self.queryBtn.clicked.connect(Dialog.queryWeather)
    . j) C' p! Z" }/ [- }# \# O
    0 G. e6 @7 p/ y& g5 j8 H        QtCore.QMetaObject.connectSlotsByName(Dialog); S' J7 @4 G6 h: u' w

    3 W& i% b* M2 @# e  N9 r5 s1 \    def retranslateUi(self, Dialog):
    ' ^8 y& a- Q) _+ F! j' E" v# d7 e, [" ]3 Z
            _translate = QtCore.QCoreApplication.translate
    ; O; [% A7 y8 G- Q% d- T& k7 v" C& V: J, r
            Dialog.setWindowTitle(_translate("Dialog", "Dialog"))% [: {9 m! r- T: v

    0 G# X& n. K" T7 U, l% _2 F; t0 h. s        self.groupBox.setTitle(_translate("Dialog", "城市天气预报"))+ s2 H7 p& u4 g
    ; r7 ~5 m8 i9 \* u6 M: l
            self.label_2.setText(_translate("Dialog", "城市")): }7 U) F% \! }7 E" K. [0 [
    7 T  N' a- r. W9 C
            self.comboBox.setItemText(0, _translate("Dialog", "北京"))
    : w8 x4 z* G/ S5 N' ~( K5 p0 a! l5 @( A5 B& C  E4 ^7 v$ P! |
            self.comboBox.setItemText(1, _translate("Dialog", "苏州"))4 `: I0 w0 T: V" o0 P  Q5 Y

    * z' n2 }  B$ Y& `( I7 }* A+ g        self.comboBox.setItemText(2, _translate("Dialog", "上海"))
    ; r4 Q$ U4 s1 e6 o* b2 s: Z# l& P4 k8 |' U% ?3 o) q
            self.queryBtn.setText(_translate("Dialog", "查询"))
    ' I/ @5 g2 j& d6 s  X2 v8 H1 c* k3 e- a: M
            self.clearBtn.setText(_translate("Dialog", "清空"))' }$ I& t5 G4 O; m

    . S/ N# M  q; s) ]2 X  c3、调用MainDialog6 ?- c& r8 {0 {
    4 p5 d( |- x7 F1 [& n1 e
    在MainDialog中调用界面类Ui_Dialog,然后在其中中添加查询天气的业务逻辑代码,这样就做到了界面显示和业务逻辑的分离。新增demo.py文件, 在MainDialog类中定义了两个槽函数queryWeather()和clearText(),以便在界面文件Weather.ui中定义的两个按钮(queryBtn 和clearBtn) 触发clicked 信号与这两个槽函数进行绑定。
    . Q5 y/ [3 q# }; z( U
    ' d2 g4 `8 T2 l, o  Y; }完整代码如下:
    2 C6 X6 L( {5 _. L- b( l/ p5 |* K: Y" {) c2 e1 P  k; A
    import sys
    - l, \+ k1 `/ {3 O/ e
      G- M3 [0 v* Z5 L* }7 limport Weather! n- j1 A8 C: U7 T2 m

    8 p& }4 p2 B) _: ?! ?* ofrom PyQt5.QtWidgets import QApplication, QDialog
    ' Y" G) q& Z3 S* r$ M* \' K
    0 N- q# X" B. e, zimport requests
    3 M( i& m9 E- ]5 C' f
    9 E, }( X) B5 Z/ C$ G% cclass MainDialog(QDialog):- o7 O7 J& Q- G: R# P6 `
    ; Y2 V7 F# Z* V; M% ]
        def __init__(self, parent=None):
    / i/ m4 Q  K  r3 P5 B
    2 o* Y4 u" Z  x; s        super(QDialog, self).__init__(parent)4 Z/ A% g# t% B9 W2 A$ o/ C
    : U" E4 ?) ]- u
            self.ui = Weather.Ui_Dialog()/ f8 e7 M/ c8 y1 P
    0 Y) |- p; U8 b) e" e
            self.ui.setupUi(self)* m* n* N( n& O# W! c
    " J4 k5 `% M) C. l
        def queryWeather(self):
    5 ~; C9 `- _& j
    ; A: L9 d! Y" S2 N, q8 X        cityName = self.ui.comboBox.currentText()
    1 {; s# H4 k# f' {0 ~% h' J, G) }- L8 B: Y
            cityCode = self.getCode(cityName)
    / }' ~( F0 i+ l& M
      ~, D4 E" A8 u. p3 o' [        r = requests.get(
    1 m- ~$ r" t- y+ Z6 D' P, W! A7 d& F- J
                "https://restapi.amap.com/v3/weather/weatherInfo?key=f4fd5b287b6d7d51a3c60fee24e42002&city={}".format(
    7 I& [7 i$ L1 F- x' X( w4 n* \: W3 p4 q. f& T7 Z. x
                    cityCode))
    0 v' ?3 a9 W0 `8 P( S
    2 I6 N! t0 l% ]% _        if r.status_code == 200:$ ?: T, {7 H% k- Q$ I1 N
    ( G7 F6 Z( G. j) [% l' u! h
                data = r.json()['lives'][0]
    & @; H$ e8 x! G7 S
    ; n; T& l( p8 E$ T5 s) s! j) Q: m( V$ c            weatherMsg = '城市:{}\n天气:{}\n温度:{}\n风向:{}\n风力:{}\n湿度:{}\n发布时间:{}\n'.format(
    / ^, ?4 U9 w1 v3 k7 v# T4 s" k% X0 e: f4 P
    5 C$ |' Q9 b) _1 h1 }( T                data['city'],# n' l- @" Z, q% V
    * \% L: l8 J. A% z/ o' G' ^
                    data['weather'],
    6 j5 S, `) ~5 w  k" D. t) Z' y4 I' z
                    data['temperature'],' e3 Q& j6 Z6 j0 w0 y1 ^3 b

    1 D1 i6 z* c5 Y3 c0 [                data['winddirection'],0 M+ @- l! L+ \8 k! w8 E+ x" H; G
    6 q, l2 c2 N6 q
                    data['windpower'],; t- T6 `0 T0 H- Q2 @% f

    / K+ W9 R/ m$ [- l6 v0 D2 ^8 e/ i                data['humidity'],
    5 t0 C6 v; r& {9 v3 Z3 y3 K3 z5 i% F/ M& N! b
                    data['reporttime'],) |* h8 `0 U6 G# h& H) Q
    $ \/ [/ b" F( x( f. Q
                )
    ' h) c1 a; z# H) J, t6 u5 q! [0 P
            else:: R$ T5 s7 _3 Y/ ?% N

    $ A; B, A: X+ b# B4 P; n  r            weatherMsg = '天气查询失败,请稍后再试!'
    9 p: m( Z/ q) M8 G) u9 s7 D( s! R% m* |: N5 U1 G( U0 h
            self.ui.textEdit.setText(weatherMsg)& p6 b# y1 H- T3 X/ y
    / l& [% a5 \6 g# [# Q
        def getCode(self, cityName):- N! T- T/ i1 t3 {2 u6 v

    - a# d6 g. z- Z  b# w6 B: m$ l) `% ]        cityDict = {"北京": "110000",. F" J: K& L0 F$ f9 O
    7 \* h0 }# d  i" J
                        "苏州": "320500",
    9 U, C6 @- Q& C0 n9 |
    ' \& O5 m0 A. X4 n                    "上海": "310000"}
    5 [1 X, z. Y" z6 L+ ~* \9 o' u! [9 }
    - T/ y* A1 h  {8 w        **return** cityDict.get(cityName, '101010100')
    5 v8 L6 R0 Q8 P; }2 H! j
      ^* A$ g# X  M$ z    def clearText(self):5 h1 m* c3 P; }8 R) L
    7 Q3 B2 ]! }2 R& O
            self.ui.textEdit.clear()* a+ i2 A+ S  N( {6 o) ]
    5 p5 m/ J" l, ?  C8 s& P
    if __name__ == '__main__':
    ) Z  A2 Y' K- ^# [0 m7 X- ], ]3 _" `/ q
        myapp = QApplication(sys.argv)
      t6 j3 W4 S$ s$ l' J, U0 x( G# K6 z+ |' G
        myDlg = MainDialog()! x) V, ?* n( a8 g
    / `! C. M: L+ T# [! j
        myDlg.show()
    , I8 W3 k( f+ q
    0 V1 |8 ]% k3 a    sys.exit(myapp.exec_()), u8 E' V8 l, g( l' n/ P
    5 t/ d- e! m( X: M+ v! c* Q
    运行demo.py并执行查询后的效果:4 ]( K; t; L% z. G: k
    + h( P) Z/ v  W4 G

    . w0 c* p  l1 b2 Y  c! [9 \* j; d8 K; n8 l" y) D% C. T
    4、将代码打包成exe文件
    7 l7 p) h% _/ N* o. B: K9 G
    $ P9 T" U9 Q1 L9 u$ {将.py文件打包成可执行的exe在Python中称为freezing,常用的工具有:PyInstaller, py2exe, cx_Freeze, bbfreze, py2app等。功能对比:
    ' P( o3 E) ]0 w  D' e0 D/ a4 K7 v: b; p. I

    ( G  s3 \1 ]! ~, f) n/ E/ o! Y5 ?3 {' g5 @- ?6 v- c% @
    py2exe:软件更新已经不活跃,因此也就略过。# s  L; @; c+ b5 U1 M& H% l" q, P
    , c. `+ m2 l' a& _" V2 n2 c/ {) Y# X& e( Z
    pyinstaller:明确支持win8、win10、理论上支持win7,,支持apple Macos, linux。pyinsaller可以打包成文件夹形式内含exe入口执行文件的形式,也可以是一个单独的exe文件。
    9 [/ l+ o7 N# `! V* a9 i0 Z9 S1 J0 d6 q" N& H) W1 P
    fbs:基于PyInstaller,使用起来更加方便
    $ W+ y% a  e, q, g) ]3 M. I) E
    5 U" K7 f7 y0 M: \这里选择了fbs来打包。fbs的安装方法:+ p# j/ q/ q+ B4 Y+ J

    0 \# }, N  b5 H7 t# Q6 bpip install fbs5 c8 }4 b2 G4 t
    使用方法,在命令行中输入:
    & A0 W; |& o4 V3 A! s2 k2 u5 x+ p6 g+ I8 x
    fbs startproject* i( P& q9 E+ ~* H$ v# i. s/ r' A
    执行完成后需要输入一些APP的名称等。完成后会生成如下目录:
    - M+ ^5 K6 m, K; R* E
    + U( `" z2 C* ?/ ^& Q5 V0 G' y0 Z3 _4 h8 t& O& g

    5 c. Y0 r! Q3 I7 l( A3 ?+ t将刚才编写的PyQt5的代码(demo.py和Weather.py)拖到src/main/python文件夹下,删除原有的main.py,并将demo.py修改为main.py。然后打开 main.py,在文件头部添加如下代码:2 B: U) w+ [( A2 m: A

    $ f. c' x! S& i* F& Y* e' e' ?from fbs_runtime.application_context.PyQt5 import ApplicationContext
    - c+ [8 A5 m: y( i; v, J6 ^```
    ' `& C1 |& R/ C; l# R9 p  i完成后执行:
    % U* O5 p- I( _& D' {/ M```% A( N, [4 d# s, d- I/ M3 z  }4 I
    fbs freeze
    % ~9 L9 T! e# {) ?% U1 M```
    + X8 P( O" r& K) J8 S5 @即可实现打包。生成的exe可执行文件在\target\MyApp文件下。* J) \# n% N5 G7 g3 l$ q. m

    3 Z$ H$ `3 p% _————————————————
    - O, _- A3 \& _; L+ [! N, [- k版权声明:本文为CSDN博主「宋宋讲编程」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
    " x5 |- I* Z7 [5 `原文链接:https://blog.csdn.net/qiqi1220/article/details/126289667
    $ r4 P1 ?4 q  d) s  B. j# A* Y" d" F; S
    & G+ e& c) N: _
    : e. u8 p6 f" c, ~3 g. ^
    1 S- J1 e; M! W& B+ @" f; W$ ?# d! m
    zan
    转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
    您需要登录后才可以回帖 登录 | 注册地址

    qq
    收缩
    • 电话咨询

    • 04714969085
    fastpost

    关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

    手机版|Archiver| |繁體中文 手机客户端  

    蒙公网安备 15010502000194号

    Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

    GMT+8, 2026-6-14 03:27 , Processed in 0.433367 second(s), 51 queries .

    回顶部