QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2195|回复: 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使用指南!
      N/ s5 u( b- @. b1 I9 v2 \0 n' f, ], w
    使用Python开发图形界面的软件其实并不多,相对于GUI界面,可能Web方式的应用更受人欢迎。但对于像我一样对其他编程语言比如C#或WPF并不熟悉的人来说,未必不是一个好的工具。
    # p9 L2 g; w1 \( S; @5 k$ r5 S+ {  T8 [" }
    常见GUI框架! u3 t/ v7 X3 z7 s

    3 o. c$ ?8 ]0 `1 y) M+ w" g) g. Z( GPyQt5:Qt是一个跨平台的 C++图形用户界面库。QT一度被诺基亚拥,后出售给芬兰的软件公司Digia Oyj。PyQt5是基于Digia公司Qt5的Python接口,由一组Python模块构成。PyQt5本身拥有超过620个类和6000函数及方法。在可以运行于多个平台,包括:Unix, Windows, and Mac OS。
    $ M2 R, ^- i8 Z* J( v' ?* v3 ~# Y) @  B# H
    Pyside6:Pyside是QT公司官方提供的Python包,上一版本为Pyside2,对应的是QT5,最新版命名规则进行了调整,更改为Pyside6,对应的是QT6版本。由于官方出品的比较看好,缺点是发布比较晚,网上的资料没有PyQt5多。1 l5 o. \0 r2 G, B' [5 M
    * y, \* w! Y. E8 N+ a/ J6 m* S4 H
    Tkinter:Python内置的GUI框架,使用TCL实现,Python中内嵌了TCL解释器,使用它的时候不用安装额外的扩展包,直接import,跨平台。不足之处在于UI布局全靠代码实现,只有15种常用部件,显示效果简陋。+ A7 n  D7 f, U  R/ V! S

    % O* R- ]* Y; qPySimpleGUI:PySimpleGUI 是 Tkinter 一层包装。使用 PySimpleGUI 实现自定义 GUI 所需的代码量要比使用 Tkinter 直接编写相同的 GUI 要少得多。
    ! N3 l& Z8 o/ X' Z# D: ^$ |9 x/ s
    WxPython:wxPython是Python语言对流行的wxWidgets跨平台GUI工具库的绑定。用得比较广泛,跨平台,C++编写,文档少,用户可能就需要根据编程内容对不同平台中的GUI代码做一些调整。遇到问题不好解决,代码布局控件,不直观。
    " w1 {3 t9 w+ F$ E, Q, U! n) X" Q+ R5 ?. H; m6 m1 Q
    Wax:基于wxPython ,为克服wxPython的问题而制作的一个包。
    , o% k) U+ Q6 h9 C8 d: l
    ; b* s* D2 D) i0 W9 fKivy:主要针对多点触控程序,智能手机平板等,也可以在没有触屏功能的系统上,全平台支持(Windows, Linux, Mac OS X, Android and iOS.)使用Python和cython编写,中文支持差,需要自己下载中文库并且制定路径。, L0 d6 Q% q. G- ~) @0 P

    " m) F" V: l& }3 l- M* c7 UBeeWare:Write once. Deploy everywhere.需要与Kivy配合使用。
    8 T# A. M  |1 J' L$ v+ M' k+ s3 ~% n4 \+ E
    Toga:一个使用Python开发原生APP的GUI工具包。Toga由一个具有共享接口的基础组件库组成,以简化与平台无关的GUI开发。Toga适用于Mac OS、Windows、Linux(GTK)以及Android和iOS等移动平台。! m6 t9 Q0 F3 K9 D  W
    - h6 \2 F/ B. W6 Y$ ?6 K* {2 |* [
    Eel:一个轻量的 Python 库,用于制作简单的类似于 Electron(但是比它更轻量) 的离线 HTML/JS GUI 应用程序,并具有对 Python 功能(capabilities)和库的完全访问权限。
      Z& W9 y1 ~1 d4 i5 ^
    ' o# Q0 O6 t  G  L- A4 TFlexx:一个纯 Python 工具包,用来创建图形化界面应用程序。其使用 Web 技术进行界面的渲染。你可以用 Flexx 来创建桌面应用,同时也可以导出一个应用到独立的 HTML 文档。因为使用纯 Python 开发,所以 Flexx 是跨平台的。只需要有 Python 和浏览器就可以运行。
    9 F" I2 M1 Q; J
    ( y) G; u, i/ l9 D% T' n/ xpywebview是围绕 webview 组件的轻量型跨平台包装器(wrapper),它允许在其自己的本机 GUI 窗口中显示 HTML 内容。它使您可以在桌面应用程序中使用 Web 技术,同时尽最大可能隐藏使用浏览器构建GUI的事实。% R" ?7 U: ?5 H/ a) z# y

    0 x7 e7 V4 ?- k4 p+ G$ ~% @enaml:一种能够让你用最小的努力就可以实现高质量GUI界面的的Python框架,也是一种独特的编程语言。enaml将声明性语言与基于约束的布局系统结合在一起,使用户可以轻松地定义灵活布局的UI。enaml应用程序可以在任何支持Python和Qt的平台上运行。9 G0 `2 Z0 n2 M$ S

    , D& C& m$ d- ~个人想法:太多学不完,先学PyQt5,原因是资料多,学有余力再学pyside6,最后看下PySimpleGUI,看能否解决一些简单问题。
    2 `( u( A; t# h% O% _; p/ m" p
    : B! m. k  C) h6 c& {* d1 |8 C! MPyQt5简介
    ' l0 f8 ^/ }3 _, Y2 J7 E; W: O- u0 i6 v; B) U" P  H7 o; x0 T
    PyQt是Qt框架的Python语言实现,由Riverbank Computing开发,是最强大的GUI库之一。PyQt提供了一个设计良好的窗口控件集合,每一个PyQt控件都对应一个Qt控件,因此PyQt的API接口与Qt的API接口很接近,但PyQt不再使用QMake系统和Q_OBJECT宏。  L7 x+ x( D1 c
    2 x: D$ \* r1 s0 A
    PyQt5提供GPL版和商业版证书,自由开发者可以使用免费的GPL许可,如果需要将PyQt用于商业应用,则必须购买商业许可。# |- X' Z3 p( z& ~

    % q' J* N' Z) Q# h& }/ uPyQt5特性如下:
    8 f' o6 C7 R9 m2 z3 f0 s+ _4 G" T8 ~1 v4 s4 z1 h5 \
    基于高性能的Qt的GUI控件集。/ c& A7 `7 p8 l* X( {

    " q/ K  P8 y# s: l: L能够跨平台运行在Linux、Window和Mac OS系统上。, @& j7 D& Y6 C" m3 i
    # }3 n; V/ `3 O/ r' B/ r
    使用信号槽机制进行通信。2 `5 B: N; m2 ^- ]6 [' d

    3 f* B( z- u$ H1 c0 ^2 m! A对Qt库进行完全封装。
    ' _3 A4 k. c  h! E& {1 k8 ^
    6 I! C2 l- L( H6 y可以使用成熟的IDE进行界面设计,并自动生成可执行的Python代码。
    7 Q0 |" H5 X. h  l7 }7 z, ~
    ; r+ Q, Y5 x; c% f1 s7 V" x; B提供一整套种类齐全的窗口控件。
    3 n5 U, }# V9 ~$ ~, D' M$ l; q! s; q! l9 }! [9 f
    PyQt5是由一系列Python模块组成,有超过620个类,6000个函数和方法,主要模块如下:
    2 W. @( K! S% {% N$ {+ R
    * j, D  C  L, p- nQtCore:包含了核心的非 GUI 的功能。主要和时间、文件与文件夹、各种数据、流、URLs、mime 类文件、进程与线程一起使用。0 H( ~5 r4 v0 q+ T7 t
    0 O  D/ |; |$ K  t/ P2 n: L& b
    QtGui:包含了窗口系统、事件处理、2D 图像、基本绘画、字体和文字类。$ ^. k; y; V. p% t) M# o3 L% L

    * k2 d* ~7 t. c. M4 m+ e% YQtWidgets:包含了一系列创建桌面应用的 UI 元素。
    , E- h) x8 a/ F. R% F6 X1 d$ f- C, @* _; ~/ E, Y
    QtMultimedia:包含了处理多媒体的内容和调用摄像头 API 的类。
    ; ?# U, ~) F& `4 Z4 \; M# I7 K+ e
    QtBluetooth:包含了查找和连接蓝牙的类。1 R; K4 a) R/ Q0 s8 U1 ?
    ; T4 ^  W, n- P) ~
    QtNetwork:包含了网络编程的类,这些工具能让 TCP/IP 和 UDP 开发变得更加方便和可靠。
    0 M* I$ z4 v  ~9 s3 S& y; {$ C4 ~7 U9 u: H0 f  O
    QtPositioning:包含了定位的类,可以使用卫星、WiFi 甚至文本。
    3 r4 j* o  W. l8 M+ A& Q4 _( Q4 a# c2 X$ `5 w
    Enginio:包含了通过客户端进入和管理 Qt Cloud 的类。* _  A; u# h: b, b- I' |
    & k7 o9 e# l' E1 Y
    QtWebSockets:包含了 WebSocket 协议的类。: I, z" `; F7 V3 r4 h
    % J. D5 \- j2 L. C6 \' Q! ^6 V
    QtWebKit:包含了一个基 WebKit2 的 web 浏览器。; `  D$ C' R) y

    ' Q) [! r# Y- k6 L4 `QtWebKitWidgets:包含了基于 QtWidgets 的 WebKit1 的类。
    , e) n9 w0 o) d! @4 ?+ b8 N& f0 F+ r; t
    QtXml:包含了处理 xml 的类,提供了 SAX 和 DOM API 的工具。+ M' y7 B; ?- u' _) v) g6 [! {" j
    ; }. _* l. O& c8 b
    QtSvg:提供了显示 SVG 内容的类,Scalable Vector Graphics (SVG) 是一种是一种基于可扩展标记语言 (XML),用于描述二维矢量图形的图形格式(这句话来自于维基百科)。
      z- _1 [2 c8 T, n- h/ z7 s, Z& K1 U0 B& I: |- ^# w
    QtSql:提供了处理数据库的工具。
    * E- B* C# |1 u* I6 M9 [% E' O# r4 r% C- _0 }! [2 z
    QtTest:提供了测试 PyQt5 应用的工具。
    3 ~( k) L2 v9 c  K
    5 ?# h* `: o5 i1 a# a+ GPyQt5的安装- F2 _) Y) s; G9 B) w/ c: `
    8 z! O; ?- q( g, g4 A* w) V
    由于后期要使用fbs进行打包,fbs对Python 3.7以后的版本可能存在兼容问题,所以我选择了Python 3.6.8进行了整个环境的搭建。主要内容为:Python + PyCharm + PyQt5
    ! j& A' M, H+ L  _8 P  L0 j. y4 z+ Q7 ^
    安装PyQt5
    0 g: y1 ?! R' S& r) Y; t' |/ G( H( I0 i6 x! W3 ?- [
    pip install pyqt5. H' \3 f/ d  I$ X* |' D

    : ^5 M: r* X& m! z) Kpip install pyqt5-tools
    " Q+ k8 q7 D' R  g# O6 [6 ]其中pyqt5-tools为Qt Designer拖拽式的界面设计工具。安装过程中可能会报如下错误:
    * W# a8 a! F; V0 Y- {, o, H; i# P$ {2 J$ R! t- ]' Y6 g, c
    qt5-tools 5.15.2.1.2 has requirement click~=7.0, but you'll have click 8.0.1 which is incompatible.( |8 S& i8 w1 Q! x
    解决方案:. `0 w2 O% {8 \1 O5 r% Q  w7 l/ s( @
      T- q7 }7 R% `. D$ S7 w, o, F
    pip install click~=7.0
    " F& l' f/ A, p0 Z5 I( S* qQt Designer的配置
    # u2 A5 l' a* c/ ^% E- n  D: ^& e# r" ^0 g) f
    Qt Designer 是通过拖拽的方式放置控件,并实时查看控件效果进行快速UI设计。( Z: h) S7 A1 T! l& ~$ L

    9 X9 V" Z) ]# J0 d5 Y6 h3 O% @$ D) @

    0 L% b8 z! r; j) k5 F! U整个画面的构成:7 a! w! `; \) a  j) H1 H
      B! y$ U& }# y2 R9 M
    左侧的“Widget Box”就是各种可以自由拖动的组件2 V+ R; o; ]; ]3 G5 ^4 A& k2 V, J

    5 k. d3 f- y7 o4 R/ n, f& l$ P中间的“MainWindow – untitled”窗体就是画布0 W* m! B2 I9 I$ C

    2 O0 J9 l: x8 r0 o+ |右上方的”Object Inspector”可以查看当前ui的结构
    5 I" D' s6 K$ W$ n7 B- g' Q( B' F0 @
    % K& S. e6 B3 }' E* r2 _- R右侧中部的”Property Editor”可以设置当前选中组件的属性7 k5 n2 C# N% q  @% w# }
    8 t' n7 L2 ^" Q0 ]# U' l& ]( e
    右下方的”Resource Browser”可以添加各种素材,比如图片,背景等等
      V9 i" S. X1 m& p" k% Z+ f' V- V) C: h! m& U/ z. g) c; V3 e6 Z
    最终生成.ui文件(实质上是XML格式的文件),可直接使用,也可以通过pyuic5工具转换成.py文件。: j2 }/ z/ V6 l9 I9 m  t; M6 S  _2 t$ t
    ; Y: x2 D" t3 G
    QtDisigner配置8 o0 W1 n( \7 z! V
    $ j( }% J& ~; a9 C* z. ]7 F" L
    在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:
    ; s2 _5 U0 d7 u. l' }6 W7 E
    * s& e$ ^# ?& TName: QtDisigner
    ; ^0 W" L2 r% s& i" w! ~: r5 ]. \: }1 y. `
    Program : D:\Program Files\Python36\Lib\site-packages\qt5_applications\Qt\bin\designer.exe # 请根据实际修改! x- u' m& L6 T, Z/ g# V! N

    2 U$ c, N9 }3 [* T4 E+ oWorking directory: $FileDir$
    $ |8 _. t, B3 z6 y# Y; e3 G& sPyUIC配置) F, n# w+ ^8 d9 K# X
    4 E* ?4 n: a* [4 \8 _* i
    PyUIC主要是把Qt Designer生成的.ui文件换成.py文件。
    8 o2 F0 u+ ~$ D: p6 @# m, l( G3 \/ O" v
    在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:4 N" O5 V9 p! ^; c! }: \: B$ j. @
    7 N; F: z+ L; C+ M4 F
    Name: PyUIC
    0 N& u! l- u% s" Q. Y  f
    # i! t& L* _) ~/ ^Program : D:\Program Files\Python36\python.exe # 当前Python目录,请根据实际修改) C- b+ n- F( S! c4 G# S5 p! S, @

    2 \7 Y( J% A4 t3 o6 g7 OArguments: -m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py! \3 r; X  Q( {' w0 {' Z0 f

    $ k( x: W0 t  n4 R  n( IWorking directory: $FileDir$* }2 W. G: P7 z4 k" P6 R
    PyRCC配置
    * r* S+ r. ~. |5 ^; z9 T
    / x1 k" A# d& Y# r3 v9 ~$ fPyRCC主要是把编写的.qrc资源文件换成.py文件。在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:1 z. l$ v0 b. q7 B9 p  s
    & v: ^5 _7 N" @/ U$ R$ m
    Name: PyRCC" O9 _, W0 M6 z  z6 ?5 X

    . ~: E, g: P+ j9 R) D% |4 c0 X! _Program: D:\Program Files\Python36\pyrcc5.exe # 当前rcc工具目录,请根据实际修改$ \9 y! d7 N$ H$ n: n# D
    - q, v1 O7 d% ~
    Arguments: $FileName$ -o $FileNameWithoutExtension$_rc.py
    / E; x4 l& E$ [2 t+ e0 U. x
    4 G" o- O4 j; c  aWorking directory: $FileDir$
    7 W6 v7 P" b+ x! ZPyQt5使用示例4 Z* U4 M/ Z) Z/ x9 L8 g: H7 A
    5 J) [# b- I  b* L2 r: r8 W
    创建一个空白的界面:0 e1 q5 f* P$ f: [9 o# H8 G
    # R+ ]9 v& ]+ P4 R( A  s- ~9 X
    import sys& A4 A7 ]- W7 F& M# O; T$ T' s
    ( @% y/ F4 g3 {9 h2 ]
    from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel
    0 a+ _- I& a0 d0 S6 i3 N- B" [0 }+ c& s2 A" K' K3 `& Q
    app = QApplication(sys.argv)7 l" ]7 O! X& t+ L; Y
    1 T; X* n/ {% G$ z" W" H$ L
    win = QMainWindow()
    " x1 V) O5 c2 z2 ^7 r$ f; d* p' A) w( q8 B, _. t# k
    win.setGeometry(400, 400, 400, 300)1 \- p% a8 Z- ^& A8 Q1 f0 P  Y

    * x1 S7 `% D. S5 \+ t! f+ L- _win.setWindowTitle("Pyqt5 Tutorial")
    ) p! i+ r& m( a2 b0 s- e4 D4 l$ W, O5 a) D# O) M: R8 `( U/ x/ [
    win.show()
    7 C; P7 N3 Z7 Z/ m# o
    8 i; D( q' c: y6 E+ c8 O( Ssys.exit(app.exec_())
    * f1 V, V$ S8 R- b3 F) L) p( D& e, K" D  b6 O$ m2 w3 j0 ]
    " }& x" }: r, [# S2 V9 X

    4 {% [$ q6 ^6 ], H  R" @8 U1 ~其中:
    9 D2 W/ C: ]6 o; F' i
    : i; O$ [* F; B5 JQapplication():每个GUI都必须包含一个Qapplication,argv表示获取命令行参数,如果不用获取,则可以使用[]代替。/ J, r2 ]" ^; b8 u. x* C6 h

    " y3 K- }) }; t) ?QMainWindow():类似一个容器(窗口)用来包含按钮、文本、输入框等widgets。arg标识可以获取命令行执行时的参数。/ s2 P4 l) O( u0 h* q& ^3 R

    , J  u8 l9 v# C* }9 |; z! l0 wSetGeometry是用来定义 QMainWindow() 窗口的尺寸, 语法:setGeometry(x, y, width, height ),其中x,y为屏幕上的坐标点。3 ^1 g; p- J! r4 |* E6 ]+ \

    ! f. a# n! _  b1 _4 P+ sshow():用来显示窗口
    + X! z9 V2 G4 h/ E
    # F% q  j1 T( j  c6 _/ Y4 [exit(app.exec_()):设置窗口一直运行指导使用关闭按钮进行关闭$ {1 W; F* L. }' Z7 @- J

    0 }  t$ K! Y* Y! B5 BPyQt5支持的常见Widgets有:* `8 ^6 Z. b! ?7 @# W/ o7 y

    ) ]7 F" A) D) m) b1 |0 K* \! L5 y1 E# Z6 `+ J! b6 ]. k. N; u6 A
    * B* ]" k# a+ @2 A6 d, F3 J
    从上到下,从左到右依次为:Qlabel、QcomboBox、QcheckBox、QradioButton、QpushButton、QtableWidget、QlineEdit、Qslider、QProgressBar/ L) J8 q! ~* W+ H  Z: k- P& {
    ! p* s2 Y6 \% [
    对于使用Pyqt5设置文本内容,我们使用Qlabel:
    " ~  W; V# y; v, }6 u% Z8 B  R# |/ F) F* ]
    import sys
      x7 n# e; E# y8 l' }6 k% y0 I* E6 o& {
    from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel$ s, |  K2 w1 T% Q1 x/ k! I  p

    + i! e% V: o3 [$ rapp = QApplication(sys.argv)
    3 i+ o1 X! J% }* u9 b' I* F# ?3 E! l
    win = QMainWindow()2 D  e- C! Y) V3 l6 z! i

    ) A) o- W8 ~9 k2 N* t3 e  Lwin.setGeometry(400, 400, 400, 300)
    1 ^; r0 R7 p, V- S3 T
    8 d* Z; S/ i9 B# P3 xwin.setWindowTitle("Pyqt5 Tutorial")
    : B6 C4 r. f+ @/ ?9 u
    + H+ T4 q& r. Y7 U6 @\# Label Text2 B2 S" \) N- F$ u

    7 L9 z: Q4 _" e! k1 b0 R' dlabel = QLabel(win)8 W3 R( k3 H8 F1 Z% V

    ( t4 L9 M5 Z* ?9 ^3 k( Qlabel.resize(200, 100), z5 i9 |* G7 T; _
    - ^2 `4 I! J/ Y6 Y) ~- u
    label.setText("Hi this is Pyqt5")( X, v: ^: Y% H8 E( Z0 H) j% s1 v

    ( N% z" k2 t$ a, _6 }* E- M3 t0 alabel.move(100, 100)  K- N; Y  b: c5 v, n! \

    % R: J* P+ b: k1 m8 pwin.show()
    , ^& T, Q6 _% Q7 x/ D3 W3 F9 D. _! K4 M. V- S
    sys.exit(app.exec_())! ^& W( p/ ?% M9 T" U- b

    ( a* S* g. X: \3 H
    4 D8 q3 |3 ]6 X  p& O* a# V
    ' [8 M+ A7 s, A: o5 \按钮与事件:; h$ e+ A% q9 ^' k6 m8 f! O
    , H+ |7 j: k, |3 U. X' b$ r( T
    import sys0 V# ?0 N; o" _# ?$ t- @; L: M7 r

    6 z* E! q3 N% S, Jfrom PyQt5.QtWidgets import QApplication, QMainWindow, QPushButton+ O' R- F9 L  ^$ b

    ) O) o; O* M' O" Y, i2 }def click():
    3 O) _) V! F7 }- a1 u
    * T' N* L, w8 ~6 `- x) s/ s. G    print("Hy Button is clicked!")6 r& {7 M3 u3 D2 r
    $ E" U) E1 c! B
    app = QApplication(sys.argv)
    " X/ k# o4 C$ l- ~
    . [4 i* P3 ^. y6 x  }! c; t5 C, Vwin = QMainWindow()
    ; ]8 x0 F6 c! C, L  n) @6 J
    , r& t8 S5 f+ {! g/ p  ^win.setGeometry(400, 400, 400, 300)
    + x+ t5 u3 O5 C3 w7 a% F! s
    & j+ d- y. |0 {& Vwin.setWindowTitle("Pyqt5 Tutorial")  W( B  X* u) u2 a4 n

    ) `' }5 W  u" Q2 W1 s) x: p\# Button+ r+ W+ x* u9 l2 [! O% \! f7 m

    3 ~& F0 @2 J; P% f7 Hbutton = QPushButton(win)
    / o5 D1 V8 v) X1 E) D3 ?; s3 x8 H& h
    button.resize(200, 100)
    , W2 H3 a' d  i8 }& W$ k
    0 b$ Z5 T2 c5 U0 B9 n% hbutton.setText("Hi! Click Me")+ F% i1 |' x0 ]' [! V% f9 h- j! |7 r
    + k- h2 ]5 y' P! i
    button.move(100, 100)
      m) Y5 j; |0 s' s2 s
    6 i/ v4 }6 Y6 ^0 M# ]- Abutton.clicked.connect(click)
    * g7 V4 w. Z6 ]7 {; H. k7 \6 J0 u& m! l  }$ C. U
    win.show()) ~5 p$ p3 E, y2 e6 a  L
    8 H( E0 }8 B4 x, U: R6 p$ O
    sys.exit(app.exec_())
    ! l4 p8 N- b+ A+ ]* m# P: \* ^* q4 v) T5 L1 i

    : Q# j, c  t5 N/ ~( p6 B
    2 r4 Y2 a% e: b5 i  S+ u: ~* a! Mbutton.clicked.connect() 在按钮点击后执行特定的事件。
    * _0 g( t( R( k' t; L6 Q- H3 A$ j  C5 {" z1 z' N& {
    PyQt5实战
    * @% G' D; X% z3 E( F$ V* D" c# L/ X  M# s
    实战项目:简易的天气查询软件  Q* `% x0 Q8 h  _4 i

    2 p" G# {4 ~- H+ s1、使用Qt Designer设计一个界面; y+ }7 f$ ~4 O" K( ]; K1 g

    9 S" ^' P7 C9 h3 }9 U- F* r
    + B8 ?7 H" j8 r
    ! p& ?8 Q: E3 R+ q) }- w! X用到的控件有Button, GroupBox, Label,ComboBox,TextEdit,同时定义了两个按钮queryBtn及clearBtn,分别用来查询及清空天气数据。我们需要绑定槽函数,方法如下:: ]5 J; X0 ~2 b
    9 p; g. N  D4 b8 v) Y6 m' T
    在Qt Designer右下角选择 信号/槽编辑器,点击+号新增
    $ ~0 `  U& J- v' I/ N3 J/ o# u/ x+ P# U, E$ c
    分别选择queryBtn及clearBtn,选择信号 clicked(), 接收者 Dialog 及槽 accept(),(槽函数这里不知道如何定义,后期在代码里再进行修改)- K+ P5 C% ?/ h6 J# M

    ) g7 e! ]6 u  L% V6 \- b以上完成后保存为Weather.ui文件。0 c1 J5 [0 O0 h

    ! y* M! d! P( i# g. g2、转换.ui文件为.py文件
    # r, s4 v8 n0 L2 C
    2 \- R5 v% E) w; }PyQt5支持直接使用.ui文件:
    % z' M. z! O, h5 D: i6 r4 C+ e% t" ]3 V. i
    import sys8 A: Y* B2 b/ l4 }$ ~3 x
    - p- a8 K7 U4 b) b$ H! S
    from PyQt5 import QtWidgets, uic
    , a! m1 z( z/ K# i  W
    + i' j3 [5 s1 F8 |( w5 D7 Gapp = QtWidgets.QApplication(sys.argv)1 Q8 X) f% z1 X) j+ |- G' [

    % k9 z. y* p" \" Wwindow = uic.loadUi("mainwindow.ui")
    - M. `; N5 X. I8 q+ r* W% Q  ^* V. P9 P2 B
    window.show()
    * _$ |; ]% ~  t8 L5 Q0 t$ Y; T* M, u# @
    app.exec()
    ! O& \% ^# h$ o& L6 M9 V但是为了更好的自定义及修改上面的槽函数,可以使用External Tools – PyUIC,即可生成Weather.py,实际运行命令如下:3 j( ^+ Y9 {) O7 d" m

    " s0 \+ e' _! u9 JD:\Program Files\Python36\python.exe -m PyQt5.uic.pyuic Weather.ui -o Weather.py/ K4 W$ v, p3 g6 z+ u8 U' f$ p/ Y7 c
    其中,我们需要把两个按钮绑定的槽函数:  g" n! D$ X) X. |# e

    ( p1 }# V$ \( e( D1 E! I: \# m\# self.queryBtn.clicked.connect(Dialog.accept)9 ^, J2 n, v6 u+ i6 G

    0 h7 b2 ]5 g) w+ n- ^9 f0 A  X\# self.clearBtn.clicked.connect(Dialog.accept)' Z; T6 h' q& b0 f$ `* k

    ) [& z( J8 d" W9 ]* E( {; B\# 修改为:
    - m. E7 `6 \) S* G" n
    ; C; ]1 P7 X+ W9 x0 yself.queryBtn.clicked.connect(Dialog.queryWeather)+ M8 Z( i5 q9 N* _! t
    6 P5 D0 s) Z; A! t
    self.clearBtn.clicked.connect(Dialog.clearText)
    . F6 I# b6 r- f4 f2 F最终的Weather.py内容如下:9 d" {4 s1 a+ F
    & U3 [" ^1 @  Z5 X
    \# -*- coding: utf-8 -*-( Q5 r4 w9 w* d* W5 \

    $ T; f6 K. U8 r9 R* c" i\# Form implementation generated from reading ui file 'Weather.ui'# t3 ?% c) a  c2 q

    $ W% h$ O+ k1 B' ?5 E\#* c0 ^! P6 J4 Z! [0 x
    9 s6 X$ D7 Z& n8 l
    \# Created by: PyQt5 UI code generator 5.15.4
    8 C, _7 L; y5 I- k8 e# J# `; W; s+ T+ K4 Z$ q% j  i7 O' C' W
    \#
    2 l5 Z$ e9 B9 B: n8 }. q3 I3 v$ G6 r# I6 \' v
    \# WARNING: Any manual changes made to this file will be lost when pyuic5 is
    ( `) p; L: Z. `5 a- i$ v3 E0 Z7 {, L2 Q) n" W8 L
    \# run again.  Do not edit this file unless you know what you are doing.
    ( g0 |5 b6 ]6 m! L  h) f' P5 i4 l$ F) U2 S9 e
    from PyQt5 import QtCore, QtGui, QtWidgets
    4 D5 f3 p9 T; B8 _5 ?8 K6 K3 Y" @" r
    class Ui_Dialog(object):$ w) A; a; w  B) M
    0 Y! J5 f% t3 ?6 v' h) H( Z
        def setupUi(self, Dialog):; Q' c8 |6 p, {5 P% {" ?

    9 {2 _% P: q% ~" p        Dialog.setObjectName("Dialog")
    ! m; r( T) d/ g
    3 ^3 x2 Y0 w+ e  t% ^        Dialog.resize(600, 600)6 k. R! v0 C  W. M
    ) G) G2 x# n  l3 D5 z
            self.groupBox = QtWidgets.QGroupBox(Dialog)
    4 L4 ]% N, s5 h4 o7 N" I/ e; N  A8 b  N
            self.groupBox.setGeometry(QtCore.QRect(30, 20, 551, 511))
      [" d0 C4 S6 d* B1 s1 ]+ t4 R4 C' l6 j
            self.groupBox.setObjectName("groupBox")' A) w0 t: @- x! s$ I2 M8 i' X

    - k) ?9 k4 F& ?$ m9 ?) B) h7 W; `        self.label_2 = QtWidgets.QLabel(self.groupBox)7 T+ y1 J6 `$ [! U( _: @' J
    3 ]. }0 R4 y5 X) e+ @1 Z* K1 d
            self.label_2.setGeometry(QtCore.QRect(20, 30, 31, 16))0 B+ ?# ?  G: j- H9 p6 S; T1 ^) T
    3 i( |: F- K$ q& o2 n& U6 d* k
            self.label_2.setObjectName("label_2")
    # v* t6 D, m) V6 u1 ^/ r
    ) u/ Y4 _% W/ G$ B0 P) H3 ^* {. z        self.comboBox = QtWidgets.QComboBox(self.groupBox)7 Y2 _$ x+ x1 t7 \: t
    4 v, A) Y+ y: T. F$ ^
            self.comboBox.setGeometry(QtCore.QRect(70, 30, 87, 22))
    " U' g2 C1 v1 j/ C4 q( v- n; b9 {' ]; ?5 M1 W# Q
            self.comboBox.setObjectName("comboBox")% r* o% [9 h) g- g0 Q

    1 h1 A3 G3 @- p1 T' T6 b. e        self.comboBox.addItem("")
    ' G4 h) D7 m& e
    4 F4 u  N, ~* G        self.comboBox.addItem("")0 |  F( t9 H8 D9 I+ \  u
      ]# _1 c) Y& q# I# M$ E. c
            self.comboBox.addItem("")
    5 H, [  w% |+ \6 ]3 ?  y( s! c9 q* ?$ y, A# C
            self.textEdit = QtWidgets.QTextEdit(self.groupBox)
    8 K8 F8 |4 v# W% V3 u5 j/ k
    . L( S$ w' Z9 R: x$ d, S$ ?# v        self.textEdit.setGeometry(QtCore.QRect(20, 70, 491, 411))4 y/ J: P' t: g6 [  g' P3 s8 w8 ^

    1 G! I. d& p& x* m; r        self.textEdit.setObjectName("textEdit")
    0 X" O) H7 \/ \$ o: I0 p+ X9 q1 c# E4 A3 t- O( D
            self.queryBtn = QtWidgets.QPushButton(Dialog)
    - s! X  F; W+ P: d! Q6 @+ t4 ?5 A" [3 t
            self.queryBtn.setGeometry(QtCore.QRect(490, 560, 93, 28))
    % f9 G4 M8 p0 {4 A
    / [8 g: ^* ~# G3 Q2 `& k        self.queryBtn.setObjectName("queryBtn")# t5 z  h( [$ |! y- q7 t

    . R9 b8 n: D3 o        self.clearBtn = QtWidgets.QPushButton(Dialog)
    6 r. G: N2 B$ z+ K: }. {* s7 G3 X+ C% m6 i% Y* h- q
            self.clearBtn.setGeometry(QtCore.QRect(30, 560, 93, 28))
    9 I$ R& o5 U! }% H# T' H6 n$ k9 Y+ l8 k/ v, q: ~  d9 O' L' K! d( r
            self.clearBtn.setObjectName("clearBtn")
    7 J/ R2 A: K$ M- a' N
    $ o% f. O3 T- U% b' U; d0 U/ S        self.retranslateUi(Dialog)
    / ]/ ?0 Y2 j1 `; G1 k
    # k- s* _0 n3 V& ?8 p        self.clearBtn.clicked.connect(Dialog.clearText)
    / E6 t4 s, q* D! \
    & o8 R7 M& y# H: z; \$ a        self.queryBtn.clicked.connect(Dialog.queryWeather)
    5 i$ R0 @1 K: o& F4 a) H, ]2 Y
    9 q, U! }4 _8 c        QtCore.QMetaObject.connectSlotsByName(Dialog)
    4 {! l1 {  \6 q- y+ i7 ]* F' z* I0 J7 w  k
        def retranslateUi(self, Dialog):/ y6 ?! n$ X1 x/ `) T/ w
    + V- \2 s0 G3 G) |! i. F3 j' U
            _translate = QtCore.QCoreApplication.translate2 Y3 U$ y1 |+ X  o( x4 e+ x" l
    0 _( Q$ E. ?3 F9 k
            Dialog.setWindowTitle(_translate("Dialog", "Dialog"))
    5 ]& k% l  Z9 K+ E* z0 c5 k+ W1 a; x# E
            self.groupBox.setTitle(_translate("Dialog", "城市天气预报"))
    + s) n- x. I; s4 c, b5 U/ |
    * \% ^1 Q+ I2 Y6 G  A        self.label_2.setText(_translate("Dialog", "城市"))
    ; C# J" ~1 w0 ]: A% e8 h
    - m$ R! S' e8 d5 X% z        self.comboBox.setItemText(0, _translate("Dialog", "北京"))- H; s" s$ l: V5 G' l% a# A
    9 z8 ?7 n" X- x: Y
            self.comboBox.setItemText(1, _translate("Dialog", "苏州"))! D* G0 B$ \3 b2 Z( n; R+ P+ U
    8 j+ Y+ }+ ^% Y( C
            self.comboBox.setItemText(2, _translate("Dialog", "上海"))
    - i' X7 y) j1 h- o( Y9 U! Q' ?3 \1 n0 x, {. Q) |) i/ h( c, w
            self.queryBtn.setText(_translate("Dialog", "查询"))
    # I) i4 E2 O4 E3 e8 f
    ) m" n, F' [( _: J4 B7 p% L- P        self.clearBtn.setText(_translate("Dialog", "清空"))3 \" g1 z% P2 z' m8 D% _
    4 K: W5 @* c, H9 A( R4 m" U
    3、调用MainDialog
    : o  r, u: Z2 u8 d9 e6 H$ W/ L2 g' N
    在MainDialog中调用界面类Ui_Dialog,然后在其中中添加查询天气的业务逻辑代码,这样就做到了界面显示和业务逻辑的分离。新增demo.py文件, 在MainDialog类中定义了两个槽函数queryWeather()和clearText(),以便在界面文件Weather.ui中定义的两个按钮(queryBtn 和clearBtn) 触发clicked 信号与这两个槽函数进行绑定。
    4 u) R) Y$ T7 E# q7 I0 m7 O4 P6 E6 }6 m4 |7 p
    完整代码如下:
    " x1 l1 }, L- n( Y: Z1 h& m. k0 o# U( I7 ^' p
    import sys
    + r- x+ Q; Q8 x8 \, y1 E
    % b' |0 i. U, S6 K, Iimport Weather8 t: V% m, B2 Z

    1 [/ `& g9 j" I: a: U, pfrom PyQt5.QtWidgets import QApplication, QDialog* m# w  T& S3 R" I# N; Q: g! x

    ' K7 W$ L! H( f- q2 i5 Z* h  Ximport requests
    - O$ G5 h8 K1 u) n5 o4 D
    8 `9 d, u, J$ N1 j) Y$ Wclass MainDialog(QDialog):
    + x4 l2 P9 X$ k
    8 n4 e: D6 }: x4 D    def __init__(self, parent=None):8 }) v' n' [& u+ j; J8 D" Y$ M& m

    5 ~5 j, A+ y  k2 y8 [& ~: f        super(QDialog, self).__init__(parent)
    ( ^/ |6 W8 i+ ^! z. `# b2 K7 L- r2 A4 ~, _  v$ ~+ V4 [# ^
            self.ui = Weather.Ui_Dialog()
    0 K$ w1 h( h3 L
    : a, _. z* m) o. F6 ^  Y        self.ui.setupUi(self)8 C6 g8 m+ a& ^) {" Z
    ; ]$ e+ P+ Q8 Q: J* W- t
        def queryWeather(self):; |- f5 q; h, d& D" W7 [% J
    1 s8 A; b2 A  [
            cityName = self.ui.comboBox.currentText()
    3 S1 H0 ~- ?  k
    5 A' `& N7 p+ d6 s- a  w        cityCode = self.getCode(cityName): M4 u1 H( i! t( ]; u9 p, Q; Q
    7 ]7 l( |' j4 n& W4 H
            r = requests.get(
    , D' U# N6 C5 t9 c% s$ m
    , Q& n8 o4 Y" m* ~            "https://restapi.amap.com/v3/weather/weatherInfo?key=f4fd5b287b6d7d51a3c60fee24e42002&city={}".format(
    ) D1 P: s/ C5 m
      A8 r/ |. [% \( A$ C                cityCode))- \$ F/ q( ]+ `" Y

    1 Z6 S2 W. S/ D) j/ S. r        if r.status_code == 200:
    ' g$ N$ r* `$ s6 b* V9 j& ]" H; \% A# ^! n7 \) j
                data = r.json()['lives'][0]
    4 c! ]2 @2 K# Q
    3 m* x# X% Q) c$ h  `            weatherMsg = '城市:{}\n天气:{}\n温度:{}\n风向:{}\n风力:{}\n湿度:{}\n发布时间:{}\n'.format(2 I/ i5 f8 z- X
    # w( ]9 C% K$ {2 l
                    data['city'],
    : q1 s- k7 N: k; {; C2 ]* t7 [# s6 F1 n  }' o( v0 [
                    data['weather'],
    , [. l6 k# t5 l
    # S# u  x: ]# L; E                data['temperature'],  v1 l8 h* `: f' @
    ; h8 t: j& `. r; b4 E0 K' d  u
                    data['winddirection'],; W2 E0 G: [" A$ g, |9 C
    ' e4 B( K6 H1 h/ r
                    data['windpower'],
    1 g7 y5 I4 J! ^" x1 B" y" x# B* L& e! \4 L
                    data['humidity'],
    " ?/ [4 Y1 G/ x- m2 i
    ; T  x( t$ n/ P9 O6 N                data['reporttime'],
    9 `8 x2 J& _2 _0 P$ N: b6 d) R# b+ |% s. ^7 D* ?- U
                )
    ! r6 F+ C6 Q" e5 P2 @0 M+ S
    0 j" i6 B6 B9 m  P7 ]1 O* p        else:
    " X+ v" ^/ y3 p3 }) D# o, {/ Q; |
    ) s2 w: P- z! a* U4 ]            weatherMsg = '天气查询失败,请稍后再试!'
    - A# O9 M; a2 }4 H% @
    6 y8 M) q6 A. R        self.ui.textEdit.setText(weatherMsg)4 P" Y, }1 Z3 Y7 T. J/ m4 R

    , S' }) l, x$ i* }6 q! q+ y) C% a    def getCode(self, cityName):
    , K* I* Y/ t$ K+ o* o, I; d% F7 H" b0 @
            cityDict = {"北京": "110000",
    1 x% l- ~8 l' D/ l4 M+ ?" S8 g/ h% `  w# m
                        "苏州": "320500",; y' V" G, P& {& u: f  G

    ' H+ R- P1 B) m1 M/ t- Z0 ^                    "上海": "310000"}' I. W+ Z8 a% X% K9 n$ C# e

    * I% ~$ L! s( j! _4 P: I        **return** cityDict.get(cityName, '101010100')
    3 ?" {' f  a1 Q8 }& h# L5 }+ z$ N9 K! k0 l3 r* D, N( e
        def clearText(self):/ A1 V0 d  ]- G/ @5 L5 ~2 r6 ^
    * B. w$ g  k3 ]7 z
            self.ui.textEdit.clear()' U7 W- ?0 {' x% `6 p; }
    ! |) u2 I& A: W/ Y% \) j
    if __name__ == '__main__':
    + F) R9 f( E4 i) n1 a
    . |) N2 [, B9 B) w    myapp = QApplication(sys.argv)
    6 x. I1 m/ l5 O0 F6 _: L$ C6 q( p1 d. n# C; H4 e; c
        myDlg = MainDialog()# y( `0 Y9 \* ?

    ) b6 t1 G& F, T  j% a7 i    myDlg.show()' u& l# M7 d% k7 x6 b
    + F$ U% g% N  h4 @" s5 W
        sys.exit(myapp.exec_())7 n; t, j' @/ w# \7 m7 z# o5 w0 j
    ( O4 B+ K9 h5 j2 n
    运行demo.py并执行查询后的效果:6 e7 B% Z9 Q* Y1 ~
    ' b2 w- B+ M* P+ \+ Q2 _  H5 C; r

    ! [8 N. m! Q( G$ b8 z9 ]3 e* h3 k* I- D+ [' D* B
    4、将代码打包成exe文件% q( E: O8 L9 Y/ H: u
    ! F/ p5 }  O9 M$ X3 Y6 ?
    将.py文件打包成可执行的exe在Python中称为freezing,常用的工具有:PyInstaller, py2exe, cx_Freeze, bbfreze, py2app等。功能对比:1 J3 m  ^" T0 h; {; f" ?) U

    # \6 |  G  c( u1 Y* k: ~* _' R* {8 ]  D0 ^

    . ^0 d) x7 I' R4 a% q( }$ jpy2exe:软件更新已经不活跃,因此也就略过。
    1 K( N: z( T8 c! I2 t/ T, d( C- }4 N: E) P  g
    pyinstaller:明确支持win8、win10、理论上支持win7,,支持apple Macos, linux。pyinsaller可以打包成文件夹形式内含exe入口执行文件的形式,也可以是一个单独的exe文件。. ^$ j6 J+ `6 p, r: K9 m

    / ]! l  K! ~; n8 `2 Ifbs:基于PyInstaller,使用起来更加方便* Q% m" n8 l9 G

    9 f6 X  {; Y8 v. o5 }" I这里选择了fbs来打包。fbs的安装方法:
    " L% A; L% o# Q/ L5 r' E
    + T- i: n$ E8 g" Wpip install fbs* }$ n4 Q9 |3 _# r1 d) W3 B
    使用方法,在命令行中输入:: n4 a8 J3 B2 ]' \% u7 d0 c0 s: b
    9 |6 C0 G0 w+ T/ h& Y8 x6 }3 W
    fbs startproject
    $ R% Z( d* v4 h执行完成后需要输入一些APP的名称等。完成后会生成如下目录:
    + u0 }5 C0 V7 Y# v" |" J6 @; [+ \: y+ q+ B# R/ U' @
    " x; K: @" D  b6 `  T# O

    " e9 @  Q2 o7 ?3 u) q将刚才编写的PyQt5的代码(demo.py和Weather.py)拖到src/main/python文件夹下,删除原有的main.py,并将demo.py修改为main.py。然后打开 main.py,在文件头部添加如下代码:
    . M# V5 w! F" u! Y1 e; g" y* ?) p8 F$ B# W  T$ G  c) A0 Q. z$ H- N
    from fbs_runtime.application_context.PyQt5 import ApplicationContext
    % c8 u! n2 m4 u9 f. \  _4 _```: R; t/ z) F( F  E% l
    完成后执行:
    7 w9 e( [; n9 G" b% T```
    : a* X+ n7 y$ Sfbs freeze% g& q; H) r" W) e9 d7 |
    ```
    : ?1 Z% k% `+ p# @! Q即可实现打包。生成的exe可执行文件在\target\MyApp文件下。
    * {0 E- Q/ l- A. U% ~
    ' W" t+ j* C% [( j4 ^' z& Z, Y————————————————% {3 V& x& {/ x" I6 _( H$ V
    版权声明:本文为CSDN博主「宋宋讲编程」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
    $ q0 I1 z# c# j原文链接:https://blog.csdn.net/qiqi1220/article/details/126289667
    + D9 x! r, t  K* h6 r! B
    # S6 G" W3 H& Y( _( T  X  t1 U6 R; L) v4 D: c, S% T. N

    - G5 O6 q1 f6 w' L6 E
    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-4-22 14:06 , Processed in 0.461142 second(s), 51 queries .

    回顶部