QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2204|回复: 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使用指南!+ _8 R1 Y; O& ~! s8 O
    0 L/ C5 i% @" l
    使用Python开发图形界面的软件其实并不多,相对于GUI界面,可能Web方式的应用更受人欢迎。但对于像我一样对其他编程语言比如C#或WPF并不熟悉的人来说,未必不是一个好的工具。/ ~3 ^* J: w3 s0 N7 K* h! i- I
    7 B& b2 J! t# A  u9 A$ p
    常见GUI框架
    # h. h! L! @2 F6 Y- q2 N( _. J; V: E3 S: p- r
    PyQt5:Qt是一个跨平台的 C++图形用户界面库。QT一度被诺基亚拥,后出售给芬兰的软件公司Digia Oyj。PyQt5是基于Digia公司Qt5的Python接口,由一组Python模块构成。PyQt5本身拥有超过620个类和6000函数及方法。在可以运行于多个平台,包括:Unix, Windows, and Mac OS。9 m! w8 c1 k  W$ k  b0 s5 h1 A

    4 X$ y# z+ @" C; E2 r* @Pyside6:Pyside是QT公司官方提供的Python包,上一版本为Pyside2,对应的是QT5,最新版命名规则进行了调整,更改为Pyside6,对应的是QT6版本。由于官方出品的比较看好,缺点是发布比较晚,网上的资料没有PyQt5多。
    2 W' T4 T7 ]) W; E) L5 W* `) O9 Z' R0 u9 H! U3 n$ `+ `* U( M/ J: a
    Tkinter:Python内置的GUI框架,使用TCL实现,Python中内嵌了TCL解释器,使用它的时候不用安装额外的扩展包,直接import,跨平台。不足之处在于UI布局全靠代码实现,只有15种常用部件,显示效果简陋。" o" ^+ D- A, \: X! b6 o$ ]2 w
    3 G% O! _$ }" C' p8 t3 {$ ^7 L
    PySimpleGUI:PySimpleGUI 是 Tkinter 一层包装。使用 PySimpleGUI 实现自定义 GUI 所需的代码量要比使用 Tkinter 直接编写相同的 GUI 要少得多。
    9 }: {% O4 K7 _; v$ ^8 F( o6 Z8 K( v' ~# N
    WxPython:wxPython是Python语言对流行的wxWidgets跨平台GUI工具库的绑定。用得比较广泛,跨平台,C++编写,文档少,用户可能就需要根据编程内容对不同平台中的GUI代码做一些调整。遇到问题不好解决,代码布局控件,不直观。
    . [8 U3 q% U) c
    * ^% D# Q  Q3 _, I5 C' UWax:基于wxPython ,为克服wxPython的问题而制作的一个包。
    % _; W5 F9 G" a) K5 a2 L3 w* ]2 K5 e8 ^( v/ H0 T. W
    Kivy:主要针对多点触控程序,智能手机平板等,也可以在没有触屏功能的系统上,全平台支持(Windows, Linux, Mac OS X, Android and iOS.)使用Python和cython编写,中文支持差,需要自己下载中文库并且制定路径。, J4 k- u2 j2 }/ d2 f8 t% S

      D6 k+ h+ h7 X  w; r0 x) {+ I% c% iBeeWare:Write once. Deploy everywhere.需要与Kivy配合使用。/ w  Q. ^- u! w% V
    - S! J* C" s; m3 G
    Toga:一个使用Python开发原生APP的GUI工具包。Toga由一个具有共享接口的基础组件库组成,以简化与平台无关的GUI开发。Toga适用于Mac OS、Windows、Linux(GTK)以及Android和iOS等移动平台。
    ' A! V: d! {$ I/ E& K/ w, i1 g( V0 y, I- \# G
    Eel:一个轻量的 Python 库,用于制作简单的类似于 Electron(但是比它更轻量) 的离线 HTML/JS GUI 应用程序,并具有对 Python 功能(capabilities)和库的完全访问权限。
    2 R& y0 v, M! j( m+ G1 q, M$ _, k  W% [8 _# \
    Flexx:一个纯 Python 工具包,用来创建图形化界面应用程序。其使用 Web 技术进行界面的渲染。你可以用 Flexx 来创建桌面应用,同时也可以导出一个应用到独立的 HTML 文档。因为使用纯 Python 开发,所以 Flexx 是跨平台的。只需要有 Python 和浏览器就可以运行。: m9 k" V! F8 k, _3 X  c1 u' S
    $ o. w- ~) q2 d  |9 C
    pywebview是围绕 webview 组件的轻量型跨平台包装器(wrapper),它允许在其自己的本机 GUI 窗口中显示 HTML 内容。它使您可以在桌面应用程序中使用 Web 技术,同时尽最大可能隐藏使用浏览器构建GUI的事实。5 G- s9 T7 O$ n' d4 t% O
    . P% S' \& t8 y) @5 C' K# E
    enaml:一种能够让你用最小的努力就可以实现高质量GUI界面的的Python框架,也是一种独特的编程语言。enaml将声明性语言与基于约束的布局系统结合在一起,使用户可以轻松地定义灵活布局的UI。enaml应用程序可以在任何支持Python和Qt的平台上运行。, {) A& ~9 Y% i. D7 G* T$ `& z

    + e! w% \# [" G个人想法:太多学不完,先学PyQt5,原因是资料多,学有余力再学pyside6,最后看下PySimpleGUI,看能否解决一些简单问题。
    2 I& R7 d, G7 l4 K2 n! u% }/ E" E. E" A3 ^& `4 o  r( N4 Z
    PyQt5简介
    6 Z) N+ R& e$ `; v2 n$ ?
    0 _: I9 F. L8 uPyQt是Qt框架的Python语言实现,由Riverbank Computing开发,是最强大的GUI库之一。PyQt提供了一个设计良好的窗口控件集合,每一个PyQt控件都对应一个Qt控件,因此PyQt的API接口与Qt的API接口很接近,但PyQt不再使用QMake系统和Q_OBJECT宏。
    " v- r) T; D3 E; F! H+ c2 ~' V+ l* v  F& r1 m
    PyQt5提供GPL版和商业版证书,自由开发者可以使用免费的GPL许可,如果需要将PyQt用于商业应用,则必须购买商业许可。
    + i5 Y4 n% O, o9 y& i
    0 e# ]3 F) e7 l5 G1 N! X& mPyQt5特性如下:
    5 X) O2 `5 G! b+ ~; I6 A
    . z8 j! G) j/ Q! Z( q, @" Z( R% m& J4 N基于高性能的Qt的GUI控件集。
    3 w# k7 V8 B- R& D: A# b; l; K+ t& U7 H0 m
    能够跨平台运行在Linux、Window和Mac OS系统上。9 c2 n* J: A% Z; f$ ~2 K+ @  z, `

    0 H! m5 y+ V) ~; \% m使用信号槽机制进行通信。2 p9 \* s/ }4 J
    ' z# w: s( U: @: K/ o0 Q6 _
    对Qt库进行完全封装。
    ( [) O" K6 N! _: }6 ?/ V: E% e' C. U) o, T. D3 n0 o! V: G+ v7 r
    可以使用成熟的IDE进行界面设计,并自动生成可执行的Python代码。, d6 X& i! X; j7 _6 J
    - `9 E" Z+ y4 b
    提供一整套种类齐全的窗口控件。* h. Y- G$ n; k- J

    ' O  Y/ p; ^9 A6 N7 vPyQt5是由一系列Python模块组成,有超过620个类,6000个函数和方法,主要模块如下:+ Z) p1 d, f( |7 K$ P' v3 h( K; D
    ; O* v+ c8 c' H7 Q3 e1 W' e! T6 f+ {
    QtCore:包含了核心的非 GUI 的功能。主要和时间、文件与文件夹、各种数据、流、URLs、mime 类文件、进程与线程一起使用。* ]* }  }* v: h* z2 A1 E

    1 V2 z4 T2 X" |QtGui:包含了窗口系统、事件处理、2D 图像、基本绘画、字体和文字类。& B$ Q0 {7 ~; b  `' z+ i

    9 e- w+ V7 R7 N+ E! L, T( n! x% KQtWidgets:包含了一系列创建桌面应用的 UI 元素。, b  d4 [$ w" d+ L. F" ~

    / v# N3 [# S/ F# t$ X. v5 KQtMultimedia:包含了处理多媒体的内容和调用摄像头 API 的类。  k% l/ R2 s" Z9 r" Q$ U1 x

    3 ~. R% }0 F2 \8 P4 N* _QtBluetooth:包含了查找和连接蓝牙的类。
    - F; Q5 C9 ]- R  D% X# h, m3 ?0 a3 T' p# b4 {/ |/ Y
    QtNetwork:包含了网络编程的类,这些工具能让 TCP/IP 和 UDP 开发变得更加方便和可靠。
    - \& i5 @+ w' w% z3 m8 ?; {5 }5 `
    5 b/ T  A6 Z* g( EQtPositioning:包含了定位的类,可以使用卫星、WiFi 甚至文本。
    2 S* f# D" n" L$ K6 M& q, W9 E9 ]
    ! ^  {6 {# a) g: OEnginio:包含了通过客户端进入和管理 Qt Cloud 的类。
    ! }0 A) m2 Z6 l' ?* i9 g( b3 n* A& R9 `8 ~
    QtWebSockets:包含了 WebSocket 协议的类。
    / V! @/ S% F/ Q! `) n' j' z
    , |+ D! H; p- t" e- _QtWebKit:包含了一个基 WebKit2 的 web 浏览器。
    # N& ]) m0 e" O6 S
    1 ^; f9 u. b& D5 c) ~0 WQtWebKitWidgets:包含了基于 QtWidgets 的 WebKit1 的类。* p  j( [! C8 q. b( e
    ' V' I3 n) O6 T* ?
    QtXml:包含了处理 xml 的类,提供了 SAX 和 DOM API 的工具。
    ( {# S# j! ~+ t- y2 q/ u0 \0 b, u
    ( v1 g3 |- U0 V% l( p# J" \QtSvg:提供了显示 SVG 内容的类,Scalable Vector Graphics (SVG) 是一种是一种基于可扩展标记语言 (XML),用于描述二维矢量图形的图形格式(这句话来自于维基百科)。
    2 N8 t" T# l9 I
    " C2 Q2 N, f/ n3 ]# @QtSql:提供了处理数据库的工具。
    . o: ]( N5 \$ b1 m: G/ G( d' V, A$ t( D* j! G5 V0 A. t$ n0 t' `
    QtTest:提供了测试 PyQt5 应用的工具。
    9 Y8 ?7 w3 W& O! C4 \! a3 s* c# X) ?  R, e
    PyQt5的安装
    . {) N) Q; O$ s" [) D( R6 E  J7 H5 Y( k( p% j% {% b4 K( c9 k
    由于后期要使用fbs进行打包,fbs对Python 3.7以后的版本可能存在兼容问题,所以我选择了Python 3.6.8进行了整个环境的搭建。主要内容为:Python + PyCharm + PyQt5 1 e0 z. o& n' f0 E& j2 ~' P2 P

    ) N( W- Q; `* s安装PyQt5
    ; i9 U( d/ u6 ], ?2 g7 G- n# H7 @* g0 A3 {5 j3 `
    pip install pyqt5
    , ?' a# c# W* n
    * j/ ^# G* v( P( V0 P9 \pip install pyqt5-tools: ]7 [# Y$ J0 A' R+ _7 N8 A
    其中pyqt5-tools为Qt Designer拖拽式的界面设计工具。安装过程中可能会报如下错误:% g$ ?3 }2 C/ H3 Z- @/ m1 Y
    . j. C4 L/ ?7 K/ G: g
    qt5-tools 5.15.2.1.2 has requirement click~=7.0, but you'll have click 8.0.1 which is incompatible.
    ! N) Q$ s8 o( u! W' o) }% |解决方案:
    " X, t) y2 r- c4 ~& F/ G2 C  i8 K# ^# k' p
    pip install click~=7.0" G1 a1 `' n4 K7 m
    Qt Designer的配置, ^9 n* i/ G1 p1 }  h  T; i/ X6 z

    ) D$ Y% q; @6 m$ Z1 e: b; N/ jQt Designer 是通过拖拽的方式放置控件,并实时查看控件效果进行快速UI设计。
    : k5 ^, _7 J- ^+ `3 {3 @! m, `
    # a6 O. `% |. R2 l5 Z2 q6 M/ a+ Q  g4 _

    $ {1 K/ Q5 e( S; @7 U+ `9 {整个画面的构成:. f8 Y0 w6 k' b0 \+ _

    0 T) |1 N' \& Q6 y  K- R左侧的“Widget Box”就是各种可以自由拖动的组件
    $ p1 T" Q. j. Q7 s2 A. _$ l. d9 N2 c* V
    中间的“MainWindow – untitled”窗体就是画布) I; `+ ~4 k" I/ k7 Q1 V$ U

    ) L- P9 ?. V* J, T5 T% R) K右上方的”Object Inspector”可以查看当前ui的结构
    ' C1 Q' S3 n, l' F( _; h: N) B! f
    0 A5 K9 e" f1 Y6 ?% I' F" E; u. U右侧中部的”Property Editor”可以设置当前选中组件的属性2 I0 Z& z5 B6 ?3 F
    5 f' v% _( L2 @2 q4 P: Z; ?
    右下方的”Resource Browser”可以添加各种素材,比如图片,背景等等  H" V% b; o+ }2 W
    0 h1 d# W4 ?% [3 o* q
    最终生成.ui文件(实质上是XML格式的文件),可直接使用,也可以通过pyuic5工具转换成.py文件。
    , e1 @  I1 F. _* G. {. U- w* b# S5 |/ P3 t& ]5 W7 b
    QtDisigner配置
    ; i* n0 S7 @( c6 D
    0 z# V) d# U. ]- n9 Y在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:! h2 b2 i. y( k% `% R4 b

    " i7 B) q) N# r: R, `, ]Name: QtDisigner
    $ k5 _3 h+ R# I* ]* n0 R7 S' i2 w; u
    9 j! @' F5 g5 z! OProgram : D:\Program Files\Python36\Lib\site-packages\qt5_applications\Qt\bin\designer.exe # 请根据实际修改
    % h. n8 v: k9 N6 U5 J: p  ~; A2 ?) `4 p3 J
    Working directory: $FileDir$
      M" G& s. X! s/ PPyUIC配置
    / n1 R7 R: ?0 D- M# _' M
    . o3 b, L# P# u7 ^. a" \PyUIC主要是把Qt Designer生成的.ui文件换成.py文件。
    1 O) ~1 L3 Y) Y$ t$ p8 F; [  C' p
    1 q- r( [- `! ~9 I$ U) \, A在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:
    ; @$ @: m+ v# I- N! k2 a; E. X+ g# \* J. i) Q5 d5 ~
    Name: PyUIC4 H9 D+ f( O. Y" G+ a: [

    $ L. k( g8 E, Y0 O, PProgram : D:\Program Files\Python36\python.exe # 当前Python目录,请根据实际修改
    5 i6 I, e( j5 M5 t/ v, o8 C0 m# p3 b  U
    Arguments: -m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py
    - z3 u4 s! C- I5 C
    # W3 `7 d. y, T/ kWorking directory: $FileDir$
      G5 y0 |: [: _% YPyRCC配置
    ! ^; [2 ^6 a8 U- \% J! U" t% V$ ?; j% A$ U! Q
    PyRCC主要是把编写的.qrc资源文件换成.py文件。在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:* i2 m3 p6 k$ _8 O

    + X0 N) s3 m6 B. p4 T, mName: PyRCC
    0 x0 }5 Q: a& v( b. h% A3 ]& _7 {6 C8 x6 V
    Program: D:\Program Files\Python36\pyrcc5.exe # 当前rcc工具目录,请根据实际修改8 `& A, _4 w  |* [7 h% V$ J$ Y
    2 r* n5 z% X( W! C  l
    Arguments: $FileName$ -o $FileNameWithoutExtension$_rc.py
    3 u1 R, \$ p, |1 |; x- X- h* U2 C4 @& l4 f' W
    Working directory: $FileDir$
    3 S& q* A; u, G+ ^1 nPyQt5使用示例5 [) i5 _; D) T! S6 J4 q  `

    7 c2 D- C4 m- I8 {& h4 ~创建一个空白的界面:
    7 Y# {1 c$ v% a9 }
    " [' }2 W: q$ I; @0 T% Simport sys; A; ~4 B$ [3 G- C4 U- @2 u+ o
    7 q% U4 B# ]5 P+ O$ ]1 E7 Y
    from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel9 ?. d! [3 p3 l  ^: i1 I$ d

    $ d' |6 y# T" P: Vapp = QApplication(sys.argv)* R1 Z4 r8 X+ i" S& B% G: |
    7 X0 Q& @' i- W6 y
    win = QMainWindow()
    ; F5 j4 w" Q9 z' C2 _  X
    # \0 U/ E/ R( P3 f6 g& t) jwin.setGeometry(400, 400, 400, 300)
    ' l8 Y" |3 u5 |3 q# ^# t+ N; j  w  C; Y4 j. N5 k$ i. l  w
    win.setWindowTitle("Pyqt5 Tutorial")
    3 p* K. G; h4 g8 `' I( @1 Y# b% t+ h$ O: @
    win.show(). D( c0 D& V6 N& c8 s5 c' w

    8 x" J2 S% g0 H8 s8 T9 `/ h* x% tsys.exit(app.exec_())9 s3 L+ W0 K' A
    / W) z4 u* P/ A) D5 Y3 u8 g' W
    8 z" x% G$ D* M- k5 q, _# V' ^

    ! ?* ^4 u/ z2 ], M- n4 E其中:  }% a5 G; L& `3 ~" W0 ]: _' k! O
    6 f! y8 n. o" L
    Qapplication():每个GUI都必须包含一个Qapplication,argv表示获取命令行参数,如果不用获取,则可以使用[]代替。
    ! g' E+ Y+ J  P
    4 t6 ]+ }7 i! D7 i+ f2 s2 Q. \QMainWindow():类似一个容器(窗口)用来包含按钮、文本、输入框等widgets。arg标识可以获取命令行执行时的参数。
    # o! d  e) t3 n3 J0 ]; [1 |9 V  n( k# k
    SetGeometry是用来定义 QMainWindow() 窗口的尺寸, 语法:setGeometry(x, y, width, height ),其中x,y为屏幕上的坐标点。2 j) t: W8 _1 {5 u+ \. N) {
    : Q: E- J5 q! g# c
    show():用来显示窗口" f# c4 Y' I2 J
    ) B3 D) k  d' @% v9 A
    exit(app.exec_()):设置窗口一直运行指导使用关闭按钮进行关闭# l* T+ h( Z( _6 y. I9 y4 \8 D: S
    ' n; w/ m5 \  Z3 J$ q% _
    PyQt5支持的常见Widgets有:& x& w- B: U4 e' w) s
    $ o0 v  J9 j- e3 [

    $ j. V8 k+ c( P5 h' c' c5 p' q5 O( c" G  Z/ s1 u
    从上到下,从左到右依次为:Qlabel、QcomboBox、QcheckBox、QradioButton、QpushButton、QtableWidget、QlineEdit、Qslider、QProgressBar- o( f, l6 Y, m. s$ y# ]
    % @. ?2 h! z) ?: n6 L) ?( Q3 T& A
    对于使用Pyqt5设置文本内容,我们使用Qlabel:
    8 e, D7 G8 F$ _% A7 F: e
      U5 h- [6 d' `! |/ U1 T. Yimport sys# B8 F0 x8 k) p0 v

    + s, T2 G* L; b3 q; r7 Dfrom PyQt5.QtWidgets import QApplication, QMainWindow, QLabel
    ' s2 F) X: O, o( ?7 O
    ! O; S3 r8 K# aapp = QApplication(sys.argv): ^1 t# O* w" y# X( m) N
    6 C! P, ~- M! C6 c
    win = QMainWindow()
    0 i/ G& U! h) g6 p9 j' Y. X, U2 l6 I8 G# A- D1 V" K. v$ x
    win.setGeometry(400, 400, 400, 300)0 I, G2 J! f: }  q- D3 }6 g1 s
    + g% P2 M: Q! e
    win.setWindowTitle("Pyqt5 Tutorial")
    / x2 y2 f* O+ O. Z. @" w/ j3 O4 C9 L' B7 H0 q
    \# Label Text
    $ N+ V5 `* g: L) A
    & ?7 R9 E1 F( Flabel = QLabel(win)
    2 F; r1 i: }2 U( ~3 t( @& U8 T7 K/ Y5 E/ g; W" i" X
    label.resize(200, 100)$ s# A- H7 Z6 Y$ j9 S% j

    / L# }4 k( k% V2 k& |label.setText("Hi this is Pyqt5"), l  n2 v2 f* B  j9 u1 ?# M
    - i- u$ @3 o: d# Y
    label.move(100, 100)
    7 Y% [( z6 J8 w7 p6 q5 [
      s* D6 S6 g( a' T7 Q& iwin.show()/ a. R5 a$ f% \+ j  I7 Y" H- o
    ; f7 G4 U) w; M9 P7 [6 J" p
    sys.exit(app.exec_())
    5 Q. |1 i0 S4 r- t/ d$ @0 ?& u5 V
    6 k5 v9 s8 Y+ d5 ~, x9 W. @
    : G: H: ^) g# c
    按钮与事件:
      s! P7 i( ^. O' `2 x: j
    : P% \  h2 K  I8 Z3 r! v7 O+ Pimport sys8 V- \+ j% C6 Q( y5 I2 \
    # d& B- H+ ~) Q, i' V
    from PyQt5.QtWidgets import QApplication, QMainWindow, QPushButton
    6 _9 v; y' @0 G+ \3 I
    / k5 g  b! i) Q& \9 }: h& zdef click():
    . H) r" a5 R6 r0 D
    6 c, @: H2 h% H" e4 ^) r5 p    print("Hy Button is clicked!")# z) [2 {* O% s/ F# I

    7 A. i+ g  U: M' _app = QApplication(sys.argv)# f# D5 i' X" ]) o  h7 O, p
    $ d1 l9 V! |1 ~/ a* o  T
    win = QMainWindow(), ?: r* _1 N; n% x: C0 u) A/ S
    8 R/ X) L+ E) I# m
    win.setGeometry(400, 400, 400, 300)
    " k  G/ t5 C( Q( ?+ A: G! c  N% F  ~& N
    win.setWindowTitle("Pyqt5 Tutorial")6 w+ Q7 A1 `) ?* L% p

    2 w9 v9 Q% y1 f1 H+ [5 R- ]3 o\# Button
    2 H/ L. k( o+ z3 [7 N+ s1 t5 F, d% `* [5 h0 c! e5 S0 b
    button = QPushButton(win)- X( D) U: p$ e$ x- V" ~
    ( c- S2 S. u9 k0 k$ A
    button.resize(200, 100)( a- C8 h- Q+ ]0 L
    % }, y4 c+ Z4 z4 y( Q- C5 m
    button.setText("Hi! Click Me")  y  i* h4 \" G% c

    " Z4 `* ?  m/ }9 d7 b7 y" u8 Mbutton.move(100, 100)
    8 @; c0 w  l! o. |9 P: b' c
    9 O: q2 p+ }% t7 ]+ c& t  P1 i# Ebutton.clicked.connect(click)& h8 \1 Z; |0 \% o: d$ S3 b

    1 W3 }4 z6 D+ n1 V  lwin.show()+ ]0 ?+ s2 u$ Z* c& z! N  v' c
    9 V3 {7 y! K  J8 d5 ^; T
    sys.exit(app.exec_())- p4 M" Z( u) I' a' \9 i/ z3 N! O

    : _! |! U4 @2 T& j/ G; v' q" a% _, D/ G
    2 h: C9 p$ @& D9 S2 Z: v4 U2 Q
    button.clicked.connect() 在按钮点击后执行特定的事件。+ s* G" O+ E0 D" M, n

    & Y5 `$ T+ ?. X0 a4 n4 P$ PPyQt5实战
    & G& W2 R; ?3 V: q* u5 r) q* ?6 i% z: p0 l4 x& M8 L9 ~
    实战项目:简易的天气查询软件
    8 {  X$ a% d! T. Q, O+ d
    4 w# ^8 b( C+ S' Q1、使用Qt Designer设计一个界面) [) O- L8 C/ N$ }& T7 Z

    ! j8 Y+ J% q( V; M) x0 W' ]# i6 I' X( K
    . z  S. a+ f: E1 e
    用到的控件有Button, GroupBox, Label,ComboBox,TextEdit,同时定义了两个按钮queryBtn及clearBtn,分别用来查询及清空天气数据。我们需要绑定槽函数,方法如下:
    % x* Q3 M5 N3 l5 g, `& J& n- U1 Q# K
    在Qt Designer右下角选择 信号/槽编辑器,点击+号新增
    ; P  F& h% O& f- [) u0 e& r/ h+ s1 \2 i' Y6 c" U3 e
    分别选择queryBtn及clearBtn,选择信号 clicked(), 接收者 Dialog 及槽 accept(),(槽函数这里不知道如何定义,后期在代码里再进行修改)+ Q8 G0 }  v$ _- E. i
    9 m$ ?& z# k9 B4 R( D1 H; X
    以上完成后保存为Weather.ui文件。9 ?. a# G& V/ q9 Q, r: O# U

    0 a+ [% E4 n8 @4 |- H( H7 Y' O+ T/ x2、转换.ui文件为.py文件4 D7 z0 J" M7 [. H- ^! w

      E5 A! m; u( Q, |  B, tPyQt5支持直接使用.ui文件:+ D" f" ~1 _) g: I5 m# Z$ Q

    & Z7 k/ Q2 t5 P; ?import sys
    ( T* N% L  S8 N3 @( T+ s9 u: K* ?/ u0 d7 |+ z* H/ H+ m
    from PyQt5 import QtWidgets, uic
    5 K- U/ g# c7 B1 l- v/ b8 U1 a7 s" Y& x; b+ V% M$ K1 ]1 L
    app = QtWidgets.QApplication(sys.argv)
    ' x0 ?% B: o, z& m6 I( t( i- D6 W' g! N* q5 _, I
    window = uic.loadUi("mainwindow.ui")
    * q3 f9 A/ a2 K. ]0 e: \; y7 r# V, U, ^5 Q( @8 m0 j
    window.show()- Z5 T0 g1 `9 L( y
    & h- l! f; ?1 x0 ?4 C" P  T! S
    app.exec()) }7 Z1 e& R3 z1 o( m4 l2 }
    但是为了更好的自定义及修改上面的槽函数,可以使用External Tools – PyUIC,即可生成Weather.py,实际运行命令如下:
    + o8 w; m- J  F) M$ y: F+ z. h# V/ i- Z. K+ f
    D:\Program Files\Python36\python.exe -m PyQt5.uic.pyuic Weather.ui -o Weather.py
    . m* v# Z, s3 z% e: ^$ H其中,我们需要把两个按钮绑定的槽函数:. S7 ^% q  n+ a+ a- t! h' r) _
    ( l7 L/ |3 d6 V; U6 x' B5 s" v
    \# self.queryBtn.clicked.connect(Dialog.accept)
    ; u/ s, a: E; I6 r* ]* f2 N- d) b" f2 j0 N
    \# self.clearBtn.clicked.connect(Dialog.accept)
    ( s6 r- H$ W7 r! g' T4 D2 u' ?
    5 M2 o2 _4 Y, E1 ^, M" E. \\# 修改为:
    # I0 \) ]. I$ ^, s9 [1 ?1 O# s- G7 b& W5 n: y% J6 V
    self.queryBtn.clicked.connect(Dialog.queryWeather)
    - }  R9 L$ S! y/ V- n6 }2 ]& l- w1 w* l, p" ^$ W
    self.clearBtn.clicked.connect(Dialog.clearText)4 C6 [* Z% s  T2 J0 F4 \; A0 t
    最终的Weather.py内容如下:
    % j8 I9 G! [! }
    7 K2 |: ~" ?- }) p$ x- H% u# K" Z\# -*- coding: utf-8 -*-. H/ m' b8 R. P  R

      @; s- q* q, C  B8 L; ]* F5 c\# Form implementation generated from reading ui file 'Weather.ui'
    # @# |8 ]2 Q- a& t& I4 D4 D) ^0 K/ Z& Y
    \## v( ]8 Z2 B! x- |5 U/ h
      x# |1 V$ X; c; Q# z
    \# Created by: PyQt5 UI code generator 5.15.4$ c; r$ k- [, G) o
    2 g$ w, [( w( C5 F2 w  F
    \#
    " `8 N( d2 K: J# u( @) c. q  M" e% ^, {2 `# V$ Y4 S
    \# WARNING: Any manual changes made to this file will be lost when pyuic5 is
    # G/ W5 F4 q& M
    % P" R+ F9 i' s, N" {" v\# run again.  Do not edit this file unless you know what you are doing.7 M! k8 j+ S) T2 a; W# K! \
    4 [! b6 {' R' {7 i9 V+ ~9 i
    from PyQt5 import QtCore, QtGui, QtWidgets
    # \$ M4 f$ f& ?4 r, t* }. l
    5 ^( y( T' w6 b; O; W+ ~3 Sclass Ui_Dialog(object):) C, S0 Q6 m7 \) N1 Y
    6 b/ L" a9 s2 J" i/ }" ]# W
        def setupUi(self, Dialog):  }7 a, I9 I8 v- G+ Y& ?

    * |9 j0 W+ y/ ]0 f7 x/ K        Dialog.setObjectName("Dialog")
    * i! r4 y6 G4 ?" i  x+ i! w
    ' U! F9 \; R% _- S( }        Dialog.resize(600, 600)
    4 N. d5 c4 {2 D- ?: H; x, Q5 |
            self.groupBox = QtWidgets.QGroupBox(Dialog)
    5 b  o0 X( H- E4 ]7 f- h, U5 p. g" n1 t
            self.groupBox.setGeometry(QtCore.QRect(30, 20, 551, 511))7 T- Q9 X! I% _
    9 o- [2 `7 X5 U1 w
            self.groupBox.setObjectName("groupBox")
    ) s9 W$ v8 x3 X! J8 {+ r$ u: s) I' o6 @2 d' e: c* C
            self.label_2 = QtWidgets.QLabel(self.groupBox)& m5 G/ T4 m0 n) b

    9 J3 s& D  g) o: u! V# b: A% f        self.label_2.setGeometry(QtCore.QRect(20, 30, 31, 16))6 B, z1 D+ E: x* f- v3 Y2 Y
    1 i, D4 t% }+ m0 d3 Z4 }
            self.label_2.setObjectName("label_2")- `: _  E  F% n- U/ W  B0 o& x

    . k- w7 R4 Y7 L" c        self.comboBox = QtWidgets.QComboBox(self.groupBox)+ ~3 ^/ }  k  v" Q  i

    9 w$ |/ `- o* ]% w1 I        self.comboBox.setGeometry(QtCore.QRect(70, 30, 87, 22))
    ! R; h, B2 [7 X7 c; O$ g0 a
    ) y; {4 a% s# z4 y! [& s) P        self.comboBox.setObjectName("comboBox")
    5 g+ F" R8 {8 {1 }& z& ~; b5 Q& L/ k0 }& g- e: p  U
            self.comboBox.addItem("")
    2 o8 r- _* J% N6 W: y5 V1 T4 O3 \1 @$ @) A! M7 e
            self.comboBox.addItem("")# z# D- d% L( W  l

    : C. v$ r, q1 Q$ j! W: X( j* e        self.comboBox.addItem("")
    ( U2 q, y% E+ f+ K3 v
      x- T, a5 n. \1 t        self.textEdit = QtWidgets.QTextEdit(self.groupBox)
    - N1 H- Y$ L" q, y6 I0 t' r6 H* v; W  X7 e
            self.textEdit.setGeometry(QtCore.QRect(20, 70, 491, 411))
    1 N( V& s3 j% t( C6 {  i& n
    8 f1 p- W/ x. Q3 b        self.textEdit.setObjectName("textEdit")+ x3 I$ M7 J$ f8 Z- j* ^

    ; D$ B: u7 K! g& }# J) G. K# h. u: x        self.queryBtn = QtWidgets.QPushButton(Dialog)
    9 w/ |: n3 ^. w( b% N
    ' @" O- `& v, ~, T% Z        self.queryBtn.setGeometry(QtCore.QRect(490, 560, 93, 28)), w3 T& O8 y6 v
    * w7 C4 c% @/ L
            self.queryBtn.setObjectName("queryBtn")* v5 J4 S, B8 m/ u
    . ^3 Y, \. N0 m7 \7 e* [
            self.clearBtn = QtWidgets.QPushButton(Dialog)$ N$ a7 S  T; D- [' D

    $ f$ v# R$ y7 s. E0 A" h, t! N7 p        self.clearBtn.setGeometry(QtCore.QRect(30, 560, 93, 28)); l2 |4 \  C$ V8 C( ~
    ; N) S. ?" x4 d6 I: B0 Z! s
            self.clearBtn.setObjectName("clearBtn")
    8 g, m/ s1 _0 }: g" E/ f+ A5 ~
    5 y% c+ _5 P1 j- n, J5 m        self.retranslateUi(Dialog)
    / X- E4 l, C& Q& A, ^, _- h
    % o( v$ U3 y5 k+ V        self.clearBtn.clicked.connect(Dialog.clearText)' Y3 m6 p' |8 f# ~

    - h; K4 F' Y/ y' u& p" U        self.queryBtn.clicked.connect(Dialog.queryWeather)6 @0 r- H5 T  Y, M3 Q

    7 {8 w3 L+ u, E- i" C, X        QtCore.QMetaObject.connectSlotsByName(Dialog)( Z, H4 p. W9 E2 `  s( l
    , w- h" M8 e. V: Y
        def retranslateUi(self, Dialog):
    ) O7 C- X; t' A4 j" p$ Q
    - @5 h5 [# F( Q" _& |$ r        _translate = QtCore.QCoreApplication.translate
    3 Z8 D7 \: P5 u3 |2 {+ y  P  ]- z- C4 {3 a3 B$ Q  o
            Dialog.setWindowTitle(_translate("Dialog", "Dialog"))
    3 @1 K8 ~# x0 ~$ N7 s* t$ d; P4 j% S( V1 t6 c) ?/ z) E8 b
            self.groupBox.setTitle(_translate("Dialog", "城市天气预报"))
    6 Z& N9 ]. t4 [. F+ f( R% ^
      n- w" ?# l8 g8 c& K, U        self.label_2.setText(_translate("Dialog", "城市"))
    % o( L8 {: R' n/ M8 e9 q( D( D& Y0 e7 t% \( _# z7 ?
            self.comboBox.setItemText(0, _translate("Dialog", "北京"))" W* c; A: ~' w( Y! \" j2 j! ]% M
    ; d' }  S! Q$ m; m" _
            self.comboBox.setItemText(1, _translate("Dialog", "苏州"))8 C+ h8 X1 N8 b; x! r) V" D: ~

    4 H5 \! T2 b8 A9 X        self.comboBox.setItemText(2, _translate("Dialog", "上海"))
    7 G" ^% B, S  p* o# B6 q! A( x; w+ ~: [  C3 C  R4 Y
            self.queryBtn.setText(_translate("Dialog", "查询"))% M5 ]0 c. Q8 r' a" O* i
    % t4 U" m( V, H: Z
            self.clearBtn.setText(_translate("Dialog", "清空"))
    $ J5 S, ~$ U: \& m3 w9 E' P: Z1 [4 r  u, T) q
    3、调用MainDialog* u  T0 ]2 s4 [

    / ^/ i2 r8 m+ @; I1 Z9 o在MainDialog中调用界面类Ui_Dialog,然后在其中中添加查询天气的业务逻辑代码,这样就做到了界面显示和业务逻辑的分离。新增demo.py文件, 在MainDialog类中定义了两个槽函数queryWeather()和clearText(),以便在界面文件Weather.ui中定义的两个按钮(queryBtn 和clearBtn) 触发clicked 信号与这两个槽函数进行绑定。" r5 S2 `* Q" `. U! J
    6 m: p/ a# `7 t" @) r( S6 _
    完整代码如下:
    ) k+ A# ^; l0 w
    ! V% m$ B- r* g5 N/ \import sys! l3 z4 R5 l8 s7 K3 b' D- ^' K
    : ?8 U, O; I- g. F8 }* ~. l0 j! c
    import Weather
    2 n  d- Y6 r7 m; l  `' D  g+ [" {# t1 P# Z7 C+ a+ ?1 n  r# R
    from PyQt5.QtWidgets import QApplication, QDialog
    . d) }" q% y; P$ {' {1 z
    5 @7 l! u7 T, x& k2 r  ~import requests
    * ]2 o5 D7 h9 K0 K: v1 {. O
    . D) V& q3 l! y9 I" ?8 `3 Cclass MainDialog(QDialog):
    8 _/ M/ k0 G4 N, b; L, m% N  F, ?% m% M' m8 R1 T
        def __init__(self, parent=None):
    9 W, R% v0 m6 L) z2 h+ H! Q& W2 `$ {5 L( K' I0 H; d8 U1 n+ r
            super(QDialog, self).__init__(parent)
    ! R" x3 J9 P5 ?' L& g: z5 }
    # \3 |* y: g& I        self.ui = Weather.Ui_Dialog()- |# l' t5 F5 G8 i5 l
    ' ^7 @2 x- ^  L
            self.ui.setupUi(self)
    $ S8 J  B3 u7 q" ~
    ' Z) Y9 x7 }+ y$ \    def queryWeather(self):$ C' e& h5 b0 k/ k

    . X3 W% `% h3 t+ E        cityName = self.ui.comboBox.currentText()
    + H) A5 [1 q; g$ _2 D) Z0 R$ }+ v; v$ X$ ^2 p0 J( [
            cityCode = self.getCode(cityName)
    ; M5 ]& A: l! d) q& Y% J/ @9 X
    " v: f2 ?8 O1 f$ [        r = requests.get(2 w5 q" L9 v$ O4 @8 D
    , e5 W+ F5 }( x2 R0 M
                "https://restapi.amap.com/v3/weather/weatherInfo?key=f4fd5b287b6d7d51a3c60fee24e42002&city={}".format(
    & `6 I4 M& e6 v  g
    ; s! U* h: n# A4 b% w! F/ t0 E6 ]                cityCode))6 r4 ?5 v% e# C4 t

    ( ~: C9 y! |3 O6 k  m        if r.status_code == 200:9 `6 R6 l  r" L- B  e

    " l$ b, ~5 r/ F$ s; \            data = r.json()['lives'][0]
    1 i1 _0 |" ^. ^6 l/ C" Q6 P7 P
    0 a: r& S( d# F' p6 Y& h            weatherMsg = '城市:{}\n天气:{}\n温度:{}\n风向:{}\n风力:{}\n湿度:{}\n发布时间:{}\n'.format(
    3 _; G, r% Z) U1 n+ q: M
    - T& A3 \; a9 ?$ U5 {7 O2 U4 p% P                data['city'],. }# \; J6 b6 o! z: R4 U
    " b5 G/ R7 t. W. A& o
                    data['weather'],
      }' c4 d( h( t. d1 @, V" C" h0 m9 I" Q( U% R2 }: X
                    data['temperature'],
    4 Z6 C1 t0 A, s9 M4 h3 W6 ~) D
    4 f& O* ^% W9 R                data['winddirection'],5 E- N. ?! x: v% k1 H/ }/ \

    " V8 z5 R2 a( ?3 M0 e                data['windpower'],  }7 ]0 @( {; r

      T6 Z6 G+ T7 n  ~9 G' V# u$ H' Q7 M                data['humidity'],5 [# y% W5 i0 E9 t6 E$ p, Z
    " A, S* ?" w9 F9 K6 x$ i1 Y
                    data['reporttime'],5 W( q6 J& t. j6 j2 ]) m3 a

    * T. Q) B" B) l4 b7 e/ R4 f, |            )
    7 Z" `8 a% F( J: J3 j+ Y+ K! I3 s3 F  [: W  z0 B
            else:$ {# ?; M4 j: N. P% A1 K  _

    $ G2 s9 a( g0 h            weatherMsg = '天气查询失败,请稍后再试!'9 H' W% O0 F: A

    / H# K+ _7 v+ x! ?        self.ui.textEdit.setText(weatherMsg)# m& P' Y" e( I- C+ [0 B$ G7 g4 R
    % Q5 H( ]; S6 y! x
        def getCode(self, cityName):
    ( [' d2 B3 q. ^. E" o" o2 F5 o7 K5 |+ ^
            cityDict = {"北京": "110000",2 W& |6 d9 ]; ]1 F7 X
    1 l! \3 ]0 w2 T. D4 h
                        "苏州": "320500",7 K+ K* }  t0 q7 \

    & B3 q& E* }% s- ~' P                    "上海": "310000"}  u% [' Y" B) m) i9 M7 Z* ~3 P

    ; O' U# l$ }, o2 S1 w+ E4 R        **return** cityDict.get(cityName, '101010100')8 ?9 l9 x; \* k( k

    5 v* S/ B7 e# L$ G    def clearText(self):
    ; y) s6 G/ F5 E. s* d
    , ^$ c+ w2 ^. T2 F$ b; i        self.ui.textEdit.clear()$ U; e8 D  l9 I2 ], D9 N8 ?
    + b. G% I" h( V! P
    if __name__ == '__main__':
    8 m" [* j" W  t0 k" X
    ! ]4 g! Z! }5 c7 z    myapp = QApplication(sys.argv)) {# C/ f: ~' R# v

    9 e. D% v" {% }7 T) R$ z    myDlg = MainDialog()3 r. h/ X( M7 F/ x7 b- E

    - l' A7 d8 p  |1 S' t    myDlg.show()
    0 x  q( {) }  N) ]9 ^4 B2 G: t4 x0 ^
    % a7 g" Q( X% f4 F2 h    sys.exit(myapp.exec_())" @7 k5 z. o' x, r; I9 I

    ( D. }4 I4 p3 U运行demo.py并执行查询后的效果:4 h: U6 s: O: Q9 T8 T# D
    ; S6 ?* v; n1 {, O/ _* s
    5 J6 L$ v) o) b$ }) M
    + H# m: v% c& V7 u- `
    4、将代码打包成exe文件
    ' K. ~: N% E9 ?  o) e, U: \* b
    # i7 N- M1 ^$ U2 q将.py文件打包成可执行的exe在Python中称为freezing,常用的工具有:PyInstaller, py2exe, cx_Freeze, bbfreze, py2app等。功能对比:
    + x$ j( Z. d4 `3 _  b2 i- f2 J) `
    5 h0 ]9 J" h% G! c* {9 H" O- |# j

      @3 W5 Y+ Q0 f/ U& k- Y' Gpy2exe:软件更新已经不活跃,因此也就略过。" y& a# C) e) F  p: c! i, [1 h
    3 q9 I" `/ b$ q" q2 \3 z4 l) A0 W
    pyinstaller:明确支持win8、win10、理论上支持win7,,支持apple Macos, linux。pyinsaller可以打包成文件夹形式内含exe入口执行文件的形式,也可以是一个单独的exe文件。
    * L6 e1 d/ G4 Y6 a/ Y: `. T& B+ F: d8 X9 n3 g. o$ B4 V
    fbs:基于PyInstaller,使用起来更加方便0 B) h5 d, s: i( N' x

    + Y2 Q, E; b- G4 @. S! Y6 B; v3 c) e这里选择了fbs来打包。fbs的安装方法:  Z9 ^& |( r$ M/ ~

    2 ?* b5 G) `7 jpip install fbs5 |" v: \0 u& y; C$ N3 T
    使用方法,在命令行中输入:
    9 A( |' l! k& S' x+ k& c: s! K( Z( j+ D0 k, q( h
    fbs startproject
    + d3 Y) e" {: H) r4 F; V/ Q4 a4 y# M执行完成后需要输入一些APP的名称等。完成后会生成如下目录:6 `: [$ a' c" \0 e! p# G
    5 T5 a' d& `1 u" t6 N

    & O' y) C: s, J2 X. g7 G7 C2 [6 O  [. h& K! i
    将刚才编写的PyQt5的代码(demo.py和Weather.py)拖到src/main/python文件夹下,删除原有的main.py,并将demo.py修改为main.py。然后打开 main.py,在文件头部添加如下代码:5 s! n8 T. H1 w  z; k- h6 U

    / A0 s* U  H$ ?' \, `- b; ~from fbs_runtime.application_context.PyQt5 import ApplicationContext
    0 D- Y3 J8 C0 @( E```
    8 T! s! J1 W) g% }. n  d完成后执行:
    7 y0 E5 ]! h' u```( }( z# T$ c& D/ X$ I8 S1 U' j
    fbs freeze
    ) n- W! _! Q$ [' ]/ x+ F+ V```
    # G4 A% E& O. S4 Q1 N7 P: F即可实现打包。生成的exe可执行文件在\target\MyApp文件下。6 C5 D+ F8 r( s$ X* D
    2 L. x5 }' G3 ~$ G
    ————————————————1 z, ?6 v: D* l4 }2 ~: r
    版权声明:本文为CSDN博主「宋宋讲编程」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
    . ^4 t4 G, x# }* @, o) M原文链接:https://blog.csdn.net/qiqi1220/article/details/1262896673 O/ x8 Y6 L3 E! D/ g2 z# b
    0 o5 v+ s6 ?, f, d
    + S9 ]6 H% s% t7 m; t$ {9 u
    ( p$ r4 l+ I7 I, l) |6 `
    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-5-26 06:51 , Processed in 0.351560 second(s), 51 queries .

    回顶部