QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2191|回复: 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使用指南!
    , Y% f- H# z8 r4 \) \, h* e
    ! ]+ c8 z, _& ~% p使用Python开发图形界面的软件其实并不多,相对于GUI界面,可能Web方式的应用更受人欢迎。但对于像我一样对其他编程语言比如C#或WPF并不熟悉的人来说,未必不是一个好的工具。+ V  Z0 r. n( ~1 [. `: K5 D; d& J8 r
    # y2 N! c5 ^2 c3 B& _5 S
    常见GUI框架
    4 K  j) m: O3 ?; a4 G8 G7 |  {* _6 A9 n
    PyQt5:Qt是一个跨平台的 C++图形用户界面库。QT一度被诺基亚拥,后出售给芬兰的软件公司Digia Oyj。PyQt5是基于Digia公司Qt5的Python接口,由一组Python模块构成。PyQt5本身拥有超过620个类和6000函数及方法。在可以运行于多个平台,包括:Unix, Windows, and Mac OS。8 M* R; E" U$ Z' v2 ~
    ' B# L" c) j3 g$ m  F9 R
    Pyside6:Pyside是QT公司官方提供的Python包,上一版本为Pyside2,对应的是QT5,最新版命名规则进行了调整,更改为Pyside6,对应的是QT6版本。由于官方出品的比较看好,缺点是发布比较晚,网上的资料没有PyQt5多。( j) w' B. O! ]  ^

    2 u" D& e* a/ P4 J% A, {2 S7 aTkinter:Python内置的GUI框架,使用TCL实现,Python中内嵌了TCL解释器,使用它的时候不用安装额外的扩展包,直接import,跨平台。不足之处在于UI布局全靠代码实现,只有15种常用部件,显示效果简陋。
    9 K8 V2 X$ o+ m3 j( a, _7 C
    0 f1 F; Y2 W7 i; q: pPySimpleGUI:PySimpleGUI 是 Tkinter 一层包装。使用 PySimpleGUI 实现自定义 GUI 所需的代码量要比使用 Tkinter 直接编写相同的 GUI 要少得多。
    ! v0 {( d4 s) ~' }6 I$ c) e5 R% R# `
    + S' d2 y$ ^! A8 w; Y; TWxPython:wxPython是Python语言对流行的wxWidgets跨平台GUI工具库的绑定。用得比较广泛,跨平台,C++编写,文档少,用户可能就需要根据编程内容对不同平台中的GUI代码做一些调整。遇到问题不好解决,代码布局控件,不直观。! D; F  D$ V  Y7 L
    / A% L( q: \# d: }0 K, l
    Wax:基于wxPython ,为克服wxPython的问题而制作的一个包。
    " l. m# r+ [1 k, P5 u
    ! e4 l5 w5 ?& d; g7 P8 k% aKivy:主要针对多点触控程序,智能手机平板等,也可以在没有触屏功能的系统上,全平台支持(Windows, Linux, Mac OS X, Android and iOS.)使用Python和cython编写,中文支持差,需要自己下载中文库并且制定路径。0 q! y- X" l- Y: U- S; `) C( w

    1 y; j# T" p7 X0 O' Q; V' RBeeWare:Write once. Deploy everywhere.需要与Kivy配合使用。
    1 s, [4 t! M# b  Z+ T
    0 l- L  V: Z) M1 F1 p" oToga:一个使用Python开发原生APP的GUI工具包。Toga由一个具有共享接口的基础组件库组成,以简化与平台无关的GUI开发。Toga适用于Mac OS、Windows、Linux(GTK)以及Android和iOS等移动平台。
    / \. p+ {- C$ o8 [3 S4 N8 y/ o* w  r
    Eel:一个轻量的 Python 库,用于制作简单的类似于 Electron(但是比它更轻量) 的离线 HTML/JS GUI 应用程序,并具有对 Python 功能(capabilities)和库的完全访问权限。$ z; F# v6 }% Z6 V6 z

    : y7 o$ R3 G, q$ q. zFlexx:一个纯 Python 工具包,用来创建图形化界面应用程序。其使用 Web 技术进行界面的渲染。你可以用 Flexx 来创建桌面应用,同时也可以导出一个应用到独立的 HTML 文档。因为使用纯 Python 开发,所以 Flexx 是跨平台的。只需要有 Python 和浏览器就可以运行。
    7 J* G9 B9 u& Q5 o6 J7 V5 \
    9 [/ U$ O- R' g0 c6 n' J- \pywebview是围绕 webview 组件的轻量型跨平台包装器(wrapper),它允许在其自己的本机 GUI 窗口中显示 HTML 内容。它使您可以在桌面应用程序中使用 Web 技术,同时尽最大可能隐藏使用浏览器构建GUI的事实。
    / P( B! a6 ]: q5 a. W( O& g2 i, D; Z* N- F- g! C7 T
    enaml:一种能够让你用最小的努力就可以实现高质量GUI界面的的Python框架,也是一种独特的编程语言。enaml将声明性语言与基于约束的布局系统结合在一起,使用户可以轻松地定义灵活布局的UI。enaml应用程序可以在任何支持Python和Qt的平台上运行。
    9 b! m  M' K5 q. T- H; n: _" ^# r( L" @  L& T9 V: C+ q+ _; ~
    个人想法:太多学不完,先学PyQt5,原因是资料多,学有余力再学pyside6,最后看下PySimpleGUI,看能否解决一些简单问题。
    ) m5 j" n3 j$ ]& L& f% D
    & r0 O* _* H3 t0 w" p+ \2 m6 L6 [. ?3 sPyQt5简介! v" l& e6 @  B: @+ A* |2 g

    0 H3 g& M6 E# @4 h# j+ tPyQt是Qt框架的Python语言实现,由Riverbank Computing开发,是最强大的GUI库之一。PyQt提供了一个设计良好的窗口控件集合,每一个PyQt控件都对应一个Qt控件,因此PyQt的API接口与Qt的API接口很接近,但PyQt不再使用QMake系统和Q_OBJECT宏。1 M+ Y. @8 \5 |" J5 k( g

    ) b' M' ]) _5 ePyQt5提供GPL版和商业版证书,自由开发者可以使用免费的GPL许可,如果需要将PyQt用于商业应用,则必须购买商业许可。0 n4 W& ^& @+ \& M' _: Y: b* u9 y" C
    9 n6 {+ b! ]& [3 V! B
    PyQt5特性如下:
    - h4 a/ x7 X# s* U1 }. I. v4 m5 P% D: G6 @$ T/ T' O
    基于高性能的Qt的GUI控件集。
    4 A% O$ W+ z( u6 J
    1 X% {- B- u# I; ?! k6 A能够跨平台运行在Linux、Window和Mac OS系统上。- s% a! W$ A0 B, p" X* z' I$ m3 G4 z

    0 i6 Q: P0 H! H使用信号槽机制进行通信。. l- H. B( ^' s& Z. e

    % ~; R* f. X( v' Q- [: K; Z* r7 j8 ^对Qt库进行完全封装。
    1 i: x. n8 C: }$ C, z! a* _0 w8 x
    可以使用成熟的IDE进行界面设计,并自动生成可执行的Python代码。- R  ?: U8 w" q  [1 H: `
    ( _- I0 t6 A$ h6 k: i
    提供一整套种类齐全的窗口控件。
    ; u9 U# y- M  Y* |: x$ ?
    ! S5 T. ?& T& }7 \7 v) X0 Y# L: X0 XPyQt5是由一系列Python模块组成,有超过620个类,6000个函数和方法,主要模块如下:# P+ R2 ~* v2 Z  n. W, d
    ! M4 l  ~8 t) W# C" W" F
    QtCore:包含了核心的非 GUI 的功能。主要和时间、文件与文件夹、各种数据、流、URLs、mime 类文件、进程与线程一起使用。  E' X; D& I) S8 f5 r- r: w

    5 r' n% e" y5 s, S4 p  mQtGui:包含了窗口系统、事件处理、2D 图像、基本绘画、字体和文字类。
    . t% S" E% {4 T3 q. o% q' V
    6 }( o& v) M# v) w- s! c! D6 uQtWidgets:包含了一系列创建桌面应用的 UI 元素。
    : ]4 p& p  s: ~- V4 T8 C) x* [3 K" S1 m6 M
    QtMultimedia:包含了处理多媒体的内容和调用摄像头 API 的类。( k4 w0 d7 H$ S/ F
    $ `' }$ F5 y9 Y# K& |. W, D2 s! \
    QtBluetooth:包含了查找和连接蓝牙的类。
    # k) S) p- |  T+ ~  S# O
    / @( x) o9 R" U2 k: T1 V7 \8 J' N% `QtNetwork:包含了网络编程的类,这些工具能让 TCP/IP 和 UDP 开发变得更加方便和可靠。8 Q5 K, n7 Z! r7 R" j% h% g* f

    0 ]/ F* Z5 G9 Z3 G' ^- LQtPositioning:包含了定位的类,可以使用卫星、WiFi 甚至文本。
    / T; h, }7 ?7 u; u1 h, O
    5 B; p  |8 C; t6 M) ?6 SEnginio:包含了通过客户端进入和管理 Qt Cloud 的类。
    8 L: K9 P( T* x6 {, e9 f, z( O# K  e$ q) \  [- R
    QtWebSockets:包含了 WebSocket 协议的类。$ @. w  H  M& d0 W

    ' T. ^) q+ x0 Q( B9 ]: l9 n' qQtWebKit:包含了一个基 WebKit2 的 web 浏览器。# P; o  D. O; O+ B' W% h

    9 u4 W7 t4 H# xQtWebKitWidgets:包含了基于 QtWidgets 的 WebKit1 的类。
    $ V5 p9 v; N, s
    # E; O8 v6 X- G; Z" aQtXml:包含了处理 xml 的类,提供了 SAX 和 DOM API 的工具。/ }! H) l5 A/ ]. S) m$ L( F

    " X6 o0 b  h: n% x* @QtSvg:提供了显示 SVG 内容的类,Scalable Vector Graphics (SVG) 是一种是一种基于可扩展标记语言 (XML),用于描述二维矢量图形的图形格式(这句话来自于维基百科)。
    0 W' L+ ^. }: l
    : v8 `' T& `6 T' U7 J  ]QtSql:提供了处理数据库的工具。( m2 x! c. X9 x& [
    ' ?& G+ K! g  s0 b3 o
    QtTest:提供了测试 PyQt5 应用的工具。: T1 W9 D, E9 v1 C+ E) L
    3 _7 d! v  |6 v
    PyQt5的安装
    - V2 d# p( z( T6 u
    ! w+ s: h1 ?0 O; o- [: N! t& s由于后期要使用fbs进行打包,fbs对Python 3.7以后的版本可能存在兼容问题,所以我选择了Python 3.6.8进行了整个环境的搭建。主要内容为:Python + PyCharm + PyQt5
    1 g( @. h% T$ l8 x7 {% t
    0 P2 T- X( s* G. t4 U& X3 F安装PyQt5
    8 N$ A" H! ?* K; z- L, g& Q+ b5 J
    , b! N2 a0 O- x' r' r* dpip install pyqt5
    + E; I% o) N" r6 P; f  `! i" e2 m
    pip install pyqt5-tools
    0 v) ^( A; K' r+ x* g& t其中pyqt5-tools为Qt Designer拖拽式的界面设计工具。安装过程中可能会报如下错误:. a2 J$ S. m) W

    3 w) b( X. [$ Y: y' }qt5-tools 5.15.2.1.2 has requirement click~=7.0, but you'll have click 8.0.1 which is incompatible.
    / f+ C% T) p/ j3 H! w& z解决方案:
    7 y: i# A1 E/ y
    ! O  _8 U& A. `4 m) [: b4 H. h5 Apip install click~=7.0. C% c! E2 M0 s! }/ A* _' z
    Qt Designer的配置
    # I7 l7 ~& r/ }- M/ b! @
    $ z  r- _7 W& X* s3 o6 AQt Designer 是通过拖拽的方式放置控件,并实时查看控件效果进行快速UI设计。. Z) C4 o* `. a0 Q2 ~
    2 m. i* f$ }! \, }, u

    7 i7 G) ~( ^' T+ ^( v
    % Q9 J: `  s( Y( U) f: R! r( k整个画面的构成:3 u& o) Q* b' m, b- B' A7 K
    $ f$ m" G; u4 w7 V  P, n. Y
    左侧的“Widget Box”就是各种可以自由拖动的组件
    3 k8 Y- Y1 q" f5 ]# Y) Y
    7 q# i& E- n+ _7 ]3 [中间的“MainWindow – untitled”窗体就是画布( F& v, c: a) I: D, x5 d; e9 ]& C

    , F7 j# w; ~0 w9 D" k! D2 v右上方的”Object Inspector”可以查看当前ui的结构
    3 w& j$ y/ X# g8 ^9 o7 }/ y0 n2 _' a2 f. ?$ E$ d& |6 V1 A
    右侧中部的”Property Editor”可以设置当前选中组件的属性- ]& o+ r; H. I6 d6 l
    + k" \  P* W- b$ H9 Z( r3 N2 Z0 D/ v
    右下方的”Resource Browser”可以添加各种素材,比如图片,背景等等5 L  H' B! y) Q; p7 @6 M
    ) N0 O7 B4 q, w1 G
    最终生成.ui文件(实质上是XML格式的文件),可直接使用,也可以通过pyuic5工具转换成.py文件。
    2 ]9 U" Q0 F3 L  a. V9 S! N9 H/ r% U: Q
    QtDisigner配置
    / J: m9 |( B6 e+ J8 F( @& r- C7 t; ^" R( q( \1 J
    在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:+ M$ m. {& g/ E" ]1 `
    % M  T; U- m9 e) V8 Q
    Name: QtDisigner+ R/ t6 f7 y, l8 X

    % P  G1 n+ B! t" dProgram : D:\Program Files\Python36\Lib\site-packages\qt5_applications\Qt\bin\designer.exe # 请根据实际修改
    + E+ t! z$ t" \$ F- U4 D3 w0 c+ i: }4 P1 r
    Working directory: $FileDir$! |- U: F5 a7 l- L* [, N6 d* m
    PyUIC配置
    , U. F- l+ r5 V: d' j+ o& G1 F* X
    : [( F, R; F* ?. |4 {9 hPyUIC主要是把Qt Designer生成的.ui文件换成.py文件。
    0 ]/ A4 t' s7 `  h1 U
    / ], v- p- c; V9 U: `在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:( G8 U0 L( R1 i9 ~, q1 u

    $ G$ b+ M( z1 p& q2 jName: PyUIC# F8 e; B4 ?  _0 ]

    & n  r5 H& `. L7 xProgram : D:\Program Files\Python36\python.exe # 当前Python目录,请根据实际修改+ O# Y; x; j' Q) n
    . \! |2 b# g0 C% _9 k$ C' T' Q, q
    Arguments: -m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py
    6 {2 j3 y4 b+ D$ Q/ Q( {% w2 o/ p/ W" W8 F
    Working directory: $FileDir$! y9 E7 K1 P2 ~: T3 }" L
    PyRCC配置
    2 N) y% A! d. i
    ! u* _/ R- A. [9 a8 PPyRCC主要是把编写的.qrc资源文件换成.py文件。在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:
    / |1 e' R5 J& \9 L, ^$ Y9 u/ |6 G, r6 |0 h  ~7 }5 W
    Name: PyRCC
    + Z' s2 ]! s( `/ D9 P& E
    & w+ b+ A5 r  ]9 q" i5 wProgram: D:\Program Files\Python36\pyrcc5.exe # 当前rcc工具目录,请根据实际修改/ w3 ^# R8 ~0 }" l
    - ?" o0 E1 [5 v. k
    Arguments: $FileName$ -o $FileNameWithoutExtension$_rc.py6 i2 ]+ Z4 d: B$ K' K% m

    + B, y1 L9 E5 k# ~) rWorking directory: $FileDir$
    ' `3 M* e2 y; B1 ~0 _PyQt5使用示例
    5 Z8 U" m1 @5 `1 j" g2 z! S4 E/ y/ D/ Q
    创建一个空白的界面:4 d$ b5 H1 ?! N$ T* ?! T
    # k" j, e. m  ^
    import sys
    5 X7 y. _0 p8 M: N5 B" D, ~# `# t* p' {3 B: V
    from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel
    . W) }% R, g  m5 u+ ]8 G5 F- A
    " h  @! _3 d. w; ~app = QApplication(sys.argv)
    ' ?+ b- Y' i9 r& z5 \8 s
    4 w: A5 X& t; ], cwin = QMainWindow()" _) u9 z; ?9 x6 {8 E' ?% A, ]

    & X5 d% F) |6 R. M! O/ b# ~win.setGeometry(400, 400, 400, 300)
    4 V2 S: z7 Y( B( s
    / j- g. s! h3 I' I0 fwin.setWindowTitle("Pyqt5 Tutorial")8 D3 U" X' n# L/ }" i

    ( P0 k. ]* o+ Qwin.show()
    $ M6 A) {, t% @# O4 I* y+ n4 t5 M9 A# [- @, R$ y
    sys.exit(app.exec_())9 S: b9 x% [6 U; O2 V" B

    $ }: N1 f% X% ]" l! J5 G7 c
    1 Q. o; `. \/ y, p" n2 r; E- Y) t5 N9 N; U5 G% _( p4 n# S! z: v
    其中:+ j9 A4 y; c7 }3 ~; Y5 b  ~4 w

    ' D3 A6 `5 Y+ M, ?# }Qapplication():每个GUI都必须包含一个Qapplication,argv表示获取命令行参数,如果不用获取,则可以使用[]代替。, ~: G; E' u( z( x

    " B5 I& @  i( @, \QMainWindow():类似一个容器(窗口)用来包含按钮、文本、输入框等widgets。arg标识可以获取命令行执行时的参数。
    7 U% W8 v5 w  b
    $ w7 A7 _# {% a3 r" `- jSetGeometry是用来定义 QMainWindow() 窗口的尺寸, 语法:setGeometry(x, y, width, height ),其中x,y为屏幕上的坐标点。: @' [. Y4 L* L+ K6 H2 Q) }
    0 S$ j6 k% k: s* u/ o
    show():用来显示窗口
    8 c6 \8 X* D1 {3 J( P4 F- g, h
      D: X+ O- N& ?' j4 p4 M6 s6 xexit(app.exec_()):设置窗口一直运行指导使用关闭按钮进行关闭$ c0 [* m. q  X: ~0 L6 @
    3 y3 S- j! t: `
    PyQt5支持的常见Widgets有:; Z; }0 u: d9 d+ R

    2 Z% N  G+ u5 V
    5 b% M  F3 H; n/ e  m/ p0 T4 t& C6 N5 Q" V4 n; A
    从上到下,从左到右依次为:Qlabel、QcomboBox、QcheckBox、QradioButton、QpushButton、QtableWidget、QlineEdit、Qslider、QProgressBar) B/ g# F' J% E) O  F5 S

    9 x& _4 |" x% J/ a( `9 q对于使用Pyqt5设置文本内容,我们使用Qlabel:8 f3 I4 U1 r# z0 E( z1 D/ A  S
    ' u! l' P5 W9 r' ]4 o4 i
    import sys4 |" z% E. b- _7 ]
    4 a$ X7 p) X9 J$ }
    from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel
    6 }  X+ A+ r* ~2 G4 J& K6 i- n) W( V7 m2 i" |; S! O
    app = QApplication(sys.argv). A; w1 s5 q% g$ O7 e+ _8 P- c( j
    2 y: ?6 j) ?# {' Z. q- i: `0 l
    win = QMainWindow()
    0 {( V( t# p, v$ v  S+ k
    9 g7 x( u9 ?$ m. @5 N+ G* {8 o+ @1 rwin.setGeometry(400, 400, 400, 300)
    + e" A7 Y/ O% j
    2 V% E# n! v2 ]+ \, B+ ^win.setWindowTitle("Pyqt5 Tutorial")
    " W3 U( ]+ E, z4 [2 [: N; `/ o
    : x5 N' N% {6 z\# Label Text: G: U7 G4 W) `- q( Z" h) o

      X& U' U$ T& r; U7 h- a4 Mlabel = QLabel(win)
    ; b' e; D/ R/ d( b. p2 c; s
    ; g/ e% H& b0 X' Zlabel.resize(200, 100)1 c$ C; T  z* r

    * _, L, ?- {" t6 Xlabel.setText("Hi this is Pyqt5")
    ; d: Y' @6 ^; ~" Q! V* Z* ^* `
    . F, x0 B+ ^# g2 ylabel.move(100, 100)
    & `- O" \% M! [5 s" @6 s
    4 U- F# L) S- }+ W9 i, P- q2 C: L" {win.show()
    # c+ \4 m$ e6 G- ^/ B- V4 T5 P6 F( i  j
    sys.exit(app.exec_())* o/ f$ m: ~. \/ H  T; z3 ?% D
    , v8 |. p- |' H- N+ A) L
    - a; f; M$ V3 q; h
    ; A* ^" X. C4 j. y3 w, w' @8 ?
    按钮与事件:  d0 y4 M: u7 G  B/ X4 N, C6 x
    + O6 F  _, I, c: Z
    import sys
    & |3 @" [. z: s: K: `( \/ \4 g4 T: U3 x& P7 I
    from PyQt5.QtWidgets import QApplication, QMainWindow, QPushButton
    0 Y$ B1 N9 H/ D; K
    ' T' K% }0 q) O7 q! c: xdef click():
      l/ c& s7 e% g5 F$ Q. I* P. a
    1 S/ i, E) T  z' K! s7 C    print("Hy Button is clicked!")
    ) H5 \1 r# z+ k# H8 r% x
    9 f+ a3 M3 C4 R' t- n" |6 eapp = QApplication(sys.argv)
    # a: j( b: l; Z9 g+ \# g  O$ P) P- E" k* L4 z, W3 G. @
    win = QMainWindow()
    6 q* }) V* P* M) Q) \6 g3 M8 D' c% S1 O
    win.setGeometry(400, 400, 400, 300)
    * k3 W  E5 z' O0 y9 w1 T
    ' m! z% ?+ B- P5 C& Qwin.setWindowTitle("Pyqt5 Tutorial")+ C2 k0 u8 q; _1 ]
    . {7 y/ V( e8 O0 n& I
    \# Button2 m! W$ I" q$ I" }1 M7 C
    1 y# u( R8 _) Q, k! X! O
    button = QPushButton(win)
    9 d$ F6 q' |( [
    ! o* V2 `9 _; D7 P7 M+ `button.resize(200, 100)* C, x/ S$ p; ~, Z

    ' U; X# j0 j6 X4 U6 h7 j$ dbutton.setText("Hi! Click Me")
    5 ]: t, v* a! E  u& U( j0 a; N9 a5 z1 d* e
    button.move(100, 100)
    0 k" B; L0 W6 N: W3 k1 a* j3 ~  `4 C: h
    button.clicked.connect(click)
    0 i6 U; J/ l% X
    " c4 j7 E0 D0 O$ p; Pwin.show()
    " a! r8 `$ g9 F7 b6 I  i9 Z- q8 @) ?0 u/ e1 r- M
    sys.exit(app.exec_())
    # Y" ^4 J& g2 C" {
    , G+ U& e- {. I2 w3 T! q  c* k, L1 o2 \$ v* ~0 v
    " M; v# |6 R4 N' k
    button.clicked.connect() 在按钮点击后执行特定的事件。
    . Y+ s7 v2 Q, N6 `( k+ u0 K; ~' K( g
    PyQt5实战
    & l+ T! V4 A( b/ h& }
    : \2 _8 }7 L) P$ l* e) o+ P" E- _( R实战项目:简易的天气查询软件
    ) z5 [) h" Y2 F3 \/ D* e* R; G  N6 B- G) C* K
    1、使用Qt Designer设计一个界面/ n; b7 q6 f& }5 l( D
    1 T1 f2 M8 k- i
    & h5 l7 k" T4 I$ T- Y: z" Y7 H

    0 R- S  j; Q$ W  e: M用到的控件有Button, GroupBox, Label,ComboBox,TextEdit,同时定义了两个按钮queryBtn及clearBtn,分别用来查询及清空天气数据。我们需要绑定槽函数,方法如下:: t6 U; A* D5 R6 G

    + j* V* Z" ~2 o( q7 y& n$ k在Qt Designer右下角选择 信号/槽编辑器,点击+号新增2 j, o& c, ?7 O$ ]6 t0 A- \
    7 V  ^4 q7 ~$ k4 e. y: Z. g
    分别选择queryBtn及clearBtn,选择信号 clicked(), 接收者 Dialog 及槽 accept(),(槽函数这里不知道如何定义,后期在代码里再进行修改)
    : o6 ^! g. V& X5 ~
    " x# I; M  R2 i' W以上完成后保存为Weather.ui文件。
    3 E8 L4 @! \- S" s4 \+ o' @# G7 t' p  q
    2、转换.ui文件为.py文件
    * f: r2 _! s+ H1 O
    2 J& v, t0 d7 t/ w* N  UPyQt5支持直接使用.ui文件:: D) K) s% W2 y. t8 q

    , P2 R1 s9 f$ simport sys, P1 @) T8 E. I$ F8 r
    ! r8 ]3 k# ^( [! b; m* e
    from PyQt5 import QtWidgets, uic0 N7 x8 d' U/ H

    # w7 P. h* {) \$ v4 L, Vapp = QtWidgets.QApplication(sys.argv)
    % v/ I; w7 c9 t* n5 i) o" Z. V- O+ Y5 s
    window = uic.loadUi("mainwindow.ui")
    * b# c' B" G7 R- D" i& C9 D/ \% l9 x7 p* s1 |! `) }
    window.show()
    & n9 p  U% [/ e: Q# P. `. Y5 E
    $ k! q1 z5 y' k, R0 \9 wapp.exec()
    8 I% ^- U% h2 K' E3 S但是为了更好的自定义及修改上面的槽函数,可以使用External Tools – PyUIC,即可生成Weather.py,实际运行命令如下:
    * A# b/ V* f' h' S' C' x: d) p9 O
    2 q9 \1 f) m5 s. c1 [7 z' g) eD:\Program Files\Python36\python.exe -m PyQt5.uic.pyuic Weather.ui -o Weather.py) A6 P. L6 _. U: }' P& X/ G) a
    其中,我们需要把两个按钮绑定的槽函数:  t0 @" F  V6 ~" N! C( r* |

    ' I$ q* F8 d7 p, X& c9 h( T\# self.queryBtn.clicked.connect(Dialog.accept)
    ; l" f  h( a' P& w: d+ Z" H- Y3 B9 h2 R) Q4 S9 @
    \# self.clearBtn.clicked.connect(Dialog.accept)# v7 a: F6 i" J
    $ a1 ~. f: Q/ r% k% o1 o
    \# 修改为:: W+ F( Q3 r: r1 Y2 \5 N( W

    ) s0 U) i, X8 W- A7 iself.queryBtn.clicked.connect(Dialog.queryWeather), C0 i  A( q9 n4 x' o6 `: ^
    / p8 P# f/ T( |; E5 e# o4 e
    self.clearBtn.clicked.connect(Dialog.clearText)$ a' U! M% s: C$ {3 d. `
    最终的Weather.py内容如下:
    5 V- k- P; k1 ]7 @* S( d  S# F9 P! \) ^; l! l8 l" p2 K) m0 h
    \# -*- coding: utf-8 -*-  M  }$ J+ i+ a$ M: n6 f# o

    : o4 r) I, H, ]3 a\# Form implementation generated from reading ui file 'Weather.ui'
    6 @+ I) ]2 Z  v& l
    * O- s$ `+ e! C' ^% \) t\#
    ) d7 D/ f! K8 Q! S) |+ Z) x$ h" q4 j, z, P& u# U- W
    \# Created by: PyQt5 UI code generator 5.15.4
    . y: B& h: a2 s$ |( p3 u
    ! q- D0 r, f0 v# `6 i\#6 V; G+ w- M7 j( Y6 S7 [+ z. P

    0 F# Z7 n4 l. f: C# S2 b" n\# WARNING: Any manual changes made to this file will be lost when pyuic5 is! R5 Y' t; c" G( M

    3 {/ h5 s$ ^' w4 t! j6 ~\# run again.  Do not edit this file unless you know what you are doing.) c( d/ ~. M$ m- q& P* z- Y+ k* q

    4 g! e5 _7 \8 m0 |/ W3 kfrom PyQt5 import QtCore, QtGui, QtWidgets7 c+ N6 h0 ?5 y2 c- N- I7 Q0 K6 A1 r
    5 _, y( r. N. @3 _
    class Ui_Dialog(object):
    . R# E* S4 c$ R/ U7 m/ ?  G  _5 ^& h1 d/ b+ o
        def setupUi(self, Dialog):, p* D6 v. T! ^% e9 Z, x$ j, G
      _. m. @7 s) m' d9 m
            Dialog.setObjectName("Dialog")
    + A. Y! n/ f: _! n' [3 O( j4 q0 w" h- h8 J. D9 k3 m  i
            Dialog.resize(600, 600)
    9 G& x: I! d% M% r0 ~7 H) V+ {3 T& W3 D+ W
            self.groupBox = QtWidgets.QGroupBox(Dialog)
    ! |: q3 S- y1 `: }- M' A
    3 u1 r1 p6 O7 d5 `5 F: E1 U3 _  [        self.groupBox.setGeometry(QtCore.QRect(30, 20, 551, 511)); @2 ~9 {& r5 A4 n+ H' x- c3 c
    2 p: R& R. l! ]
            self.groupBox.setObjectName("groupBox")
    , L+ \6 n% d5 ^0 h& {$ V* d. b+ F" B! A
            self.label_2 = QtWidgets.QLabel(self.groupBox)5 h/ o' w0 G+ H+ \  a& V

    3 n1 N# S0 ^4 }0 v& b        self.label_2.setGeometry(QtCore.QRect(20, 30, 31, 16))! C6 y. V0 a8 l; }' Y

    " ~2 }) E" f7 {" s        self.label_2.setObjectName("label_2")
    2 h1 n4 t: s1 B" U1 ]; `' t3 z6 ]  J" O# ?9 J5 m4 ?& P. L2 y$ m& k& ^
            self.comboBox = QtWidgets.QComboBox(self.groupBox)
    # e% g$ ]# F9 H5 q; r" F
    ( I2 d+ z8 E. \( Y        self.comboBox.setGeometry(QtCore.QRect(70, 30, 87, 22))
    " i( t/ E% F; y' X0 P, H$ @3 b3 V- N+ S
            self.comboBox.setObjectName("comboBox")3 k6 I$ b# v. w, Y2 V6 P7 I
    5 h2 |2 c% y8 ~: p
            self.comboBox.addItem("")
    ( z$ |6 M. i% Y1 h4 L
    : w; `! u1 Z' L4 l9 s' l' h2 k  n        self.comboBox.addItem("")0 X0 c+ B' k0 K; \( j6 Y- N; |
    . W5 Y* V3 o$ j' n; l' L
            self.comboBox.addItem("")
    9 l' F6 \( F: |, X- U& \3 E% s- X3 Z6 i( B0 O; j7 T! J! ]2 r
            self.textEdit = QtWidgets.QTextEdit(self.groupBox)% y; T6 e# U! m* P0 X) Y7 L

    # `/ {$ N4 w6 H2 r! F6 H+ U        self.textEdit.setGeometry(QtCore.QRect(20, 70, 491, 411))
    6 g3 W8 ~; ]( Y3 m3 q9 |+ Z, V5 L" M* e6 f& l4 O( \8 D
            self.textEdit.setObjectName("textEdit")
    9 S, E; f3 e& z0 @" [: S' [4 {- k, z  d
            self.queryBtn = QtWidgets.QPushButton(Dialog). ]+ `1 d0 d; y

    * q7 T) s, ]1 e$ S; C' r        self.queryBtn.setGeometry(QtCore.QRect(490, 560, 93, 28))9 z5 E6 y  a& W/ y) l
    & _8 t2 ^! \( H$ A- u; G4 E
            self.queryBtn.setObjectName("queryBtn"): h; ~  g" D, t
    & q* F; f6 ?$ e" _) s% y% a
            self.clearBtn = QtWidgets.QPushButton(Dialog)5 z1 k/ g) p: c* A/ i

    & s1 A5 s' q; q/ z        self.clearBtn.setGeometry(QtCore.QRect(30, 560, 93, 28))4 ^9 ?5 Q4 X$ l8 G; A7 u
      w9 [! p: Z! \7 q1 D! R
            self.clearBtn.setObjectName("clearBtn")
    * w; v+ z' h% w$ H# h# N" ^5 `1 ^
    $ v- \, e: v! k. G        self.retranslateUi(Dialog); e2 G  B. ~- _: v

    # W* G7 Q" L; H& a        self.clearBtn.clicked.connect(Dialog.clearText)
      k2 b! |0 E' i' W# X
    8 Z% O9 L) }! M" l        self.queryBtn.clicked.connect(Dialog.queryWeather)* r) e0 O7 {3 {" a
    , v& V9 [0 J* L+ h% D, `9 w' a
            QtCore.QMetaObject.connectSlotsByName(Dialog)8 Z( a% n8 D& I4 ?- Z
    ! }! z& M0 r' |" `
        def retranslateUi(self, Dialog):0 V/ u2 d' w, w
    ( ]. }$ \) l: }& ^& B2 ]5 j
            _translate = QtCore.QCoreApplication.translate- ^2 c+ Z9 k9 R& p) F. I

    9 C! p& ?9 j) _$ J, \+ A        Dialog.setWindowTitle(_translate("Dialog", "Dialog"))' V# S- ^( M3 K& O% ~3 w* Z

    0 h/ ?5 N# T( ]" W( Z+ f$ l        self.groupBox.setTitle(_translate("Dialog", "城市天气预报"))* T0 p3 v3 u7 g6 E

    3 S' E% n1 K/ x: o; [, I/ K" g2 L6 O        self.label_2.setText(_translate("Dialog", "城市"))+ {$ ]! g, Z- k4 d

    ) \3 H# Y$ I* D0 V        self.comboBox.setItemText(0, _translate("Dialog", "北京"))
    1 _$ h3 S# G) x8 \
    7 Q) r; t7 P. b% @3 C) x; c5 H5 P        self.comboBox.setItemText(1, _translate("Dialog", "苏州"))/ I( B, H' ]& I+ s8 m$ J% h
    ! e" K; ~! \# O( k- Z0 ?6 p
            self.comboBox.setItemText(2, _translate("Dialog", "上海"))
    3 N: }9 g! H$ v5 C3 @1 s; x* h0 L
    7 [6 h! G/ w8 K        self.queryBtn.setText(_translate("Dialog", "查询"))
    : Y$ ?! z* ?/ p' W, h  k/ u- X
    # H  ]+ B9 Q* [0 a$ e        self.clearBtn.setText(_translate("Dialog", "清空"))% K5 D# {- F' d6 s

    0 r6 H; o6 ~' x3、调用MainDialog
    - `% t9 C3 z/ I, z* I8 y  j
    0 L' M4 F. C& m6 ^8 _7 I2 w" Y在MainDialog中调用界面类Ui_Dialog,然后在其中中添加查询天气的业务逻辑代码,这样就做到了界面显示和业务逻辑的分离。新增demo.py文件, 在MainDialog类中定义了两个槽函数queryWeather()和clearText(),以便在界面文件Weather.ui中定义的两个按钮(queryBtn 和clearBtn) 触发clicked 信号与这两个槽函数进行绑定。8 k) O1 F. w0 m3 B5 G. _9 N$ ^

    % _4 e; |( Y4 u( y7 ~1 ~! I# n完整代码如下:
    1 J4 `: C6 a% k# Y9 @
    & W- ~5 ~7 w# P( F0 H- Rimport sys
    ; b' c8 I8 d  t* _# {' m; u6 L/ J' j; C6 R" z: Y4 i8 k
    import Weather
      c  a4 O$ q8 Q$ h3 d) n2 C. ^5 A2 K( e- _# x5 j& O$ ]: O, ~1 N0 m6 G
    from PyQt5.QtWidgets import QApplication, QDialog0 E6 y. ?* s, V, Q7 `

    ' h; _1 ^3 h3 s) t+ Z$ P0 Aimport requests' P# f9 s0 Q+ `" Q4 k* e9 t: c

    1 ^0 v# ]- u9 c! L. Nclass MainDialog(QDialog):* e  Q5 R2 J3 f( u+ S5 P8 Z- h

    ' l  B1 m( P2 d; Y( r6 n    def __init__(self, parent=None):
    & F# v1 i( K4 n) X- Z  p# F( h
    1 Y/ _$ k9 z( t$ b6 C" f        super(QDialog, self).__init__(parent)0 u2 P5 V0 Q$ {6 G

    0 E# h9 h9 |* B8 w: G        self.ui = Weather.Ui_Dialog()
    " _+ |, R. n/ C
    $ p2 I- n2 \& z+ r1 R! v        self.ui.setupUi(self)! c7 q: K5 _/ ?  m  g" k: d6 M
    & G, A. b6 E! A$ P
        def queryWeather(self):
    4 j7 V) Y. Z. ^7 Y# J: m8 U/ @# D: l0 Y. @0 ^& s- D& i
            cityName = self.ui.comboBox.currentText()9 `( g: i  u: @

    0 U2 A) \, u" [9 ]        cityCode = self.getCode(cityName)
    ( g3 m7 |* Q3 s  s$ G
    ; Q( O& B" `! B- P        r = requests.get(( m/ ]3 {$ O9 S- L) `
    % Q+ B, e0 K* `+ i7 @
                "https://restapi.amap.com/v3/weather/weatherInfo?key=f4fd5b287b6d7d51a3c60fee24e42002&city={}".format(
      V9 r0 i8 V0 k) P! M# g7 z! s* T/ e
                    cityCode))
    " S1 A4 n# l: A1 q9 t1 V3 K. S
    " e; @$ j7 I& a% [0 x: A$ R/ M6 U        if r.status_code == 200:
    7 ^# x4 j: Z* y- H0 w& X+ @# g2 d2 Y  X+ F: ?
                data = r.json()['lives'][0]. p( T! a. l, a8 |+ X2 E

    ' M( j/ _) u" u* r8 z' I0 F            weatherMsg = '城市:{}\n天气:{}\n温度:{}\n风向:{}\n风力:{}\n湿度:{}\n发布时间:{}\n'.format(1 C' P$ b- ~7 M9 _$ a0 A4 T

    2 Q9 {! ?2 f, [" ]$ l- q                data['city'],
    % w, X, `7 d7 h& ]+ M+ t  @, f' r: M$ c, E  C9 B2 M" P
                    data['weather'],
    $ S3 V9 q: s2 R  y& S+ @
    % `" a& X1 c' a* Q/ R( @1 F" M# X                data['temperature'],
      N4 j- e  G- S& _4 M# o9 g
    & u+ f! f4 q7 U$ N                data['winddirection'],; [$ y& h# i: O& j; Q2 N9 H
    + g+ v* Z4 B( _3 Q% z& c- Q
                    data['windpower'],
    - L2 _: l/ L: X7 l2 s3 l3 V% ?; D, c: P$ t, D( G
                    data['humidity'],! f! y2 L, d0 _5 B

    ) k+ M  x, S/ T0 ?( E                data['reporttime'],
    7 ]* p, \! {9 Q- S, `* M$ E8 t1 e' u, H2 q6 U
                )
    & |6 P* a# h& @! m6 d$ V# z- M9 P- o2 S) e5 T6 t. x3 }
            else:* O4 u1 N% R" v- x- m2 s
    9 |, c9 l; G! K7 U" w7 e# M# W( t& g6 L
                weatherMsg = '天气查询失败,请稍后再试!'" B/ B8 n( d0 F, ?! J
      V3 L5 O, T) M, u0 i: E
            self.ui.textEdit.setText(weatherMsg)
    % m" |4 Q0 M+ T; G8 V
    # y: V5 J, L  O0 d$ ~    def getCode(self, cityName):8 c. o8 j- [1 O5 X8 ]

    4 U  V' ~# c! }) T4 D; G' C        cityDict = {"北京": "110000",$ L5 ]# t. T5 M7 B. [' H. }

    # g; A. W9 s! h& Y& N& o                    "苏州": "320500",
    6 b/ t- a- N) E2 z. ?0 ^% F5 v8 b! ]8 N1 y& [( ~
                        "上海": "310000"}
    " d1 e5 m8 h) B& B- j) \0 x. C$ t" ^  G- v1 j9 Y, ]1 o  G4 f
            **return** cityDict.get(cityName, '101010100')
    9 H0 w0 r% }3 k: A: f/ w9 t: [) ], M5 K! ]( S0 q, b& G
        def clearText(self):
    2 Q* f2 J2 V# U( _: n$ f8 W% M9 Y! Y: P* E$ e. A
            self.ui.textEdit.clear(). c, ~" P3 X" b# W; u

    7 N' O. y* i3 e! b% |( Vif __name__ == '__main__':+ W* r( J3 w# E% [$ h# X6 ^
    - y9 ?- }8 Z# Y: x( c( l$ T* _9 r
        myapp = QApplication(sys.argv)" S0 [8 V. i0 u5 P1 |: T5 I2 t0 R

    0 Y8 z% Y( {# B7 t0 B( Z    myDlg = MainDialog()5 v% k% y) M2 I0 H/ t+ A
    / I4 Z) U' f8 O3 ~5 w# Z/ R- T
        myDlg.show()
    . ~- h) I. C5 e1 D) T* g, `+ v) K( ~& F( u, Y, X
        sys.exit(myapp.exec_())
    9 D! f. d# \! x* P: }: W# k& }' X
    + v  l7 e* T- v! \运行demo.py并执行查询后的效果:/ E* ^  m; O5 Z! m& I8 C4 x

    + v8 @# V; a7 |4 d* H. A9 ]- m( o, G- v' p

    $ V6 H+ B8 R  \. Y+ u  `9 ]7 L4、将代码打包成exe文件0 F2 |7 d* L" e, I" o

      x+ N1 ^8 g5 h" @! t$ Z) T将.py文件打包成可执行的exe在Python中称为freezing,常用的工具有:PyInstaller, py2exe, cx_Freeze, bbfreze, py2app等。功能对比:6 D( {2 T7 q4 W/ G

    & c4 v. k' r. p8 E0 q2 t; |
    ! L; c- u: O6 J4 [
      p2 |- r( M2 G# a$ K5 i" Spy2exe:软件更新已经不活跃,因此也就略过。- S2 d; ^* ]. A$ M) t3 D. g
    ! ^* I7 `% Y6 l
    pyinstaller:明确支持win8、win10、理论上支持win7,,支持apple Macos, linux。pyinsaller可以打包成文件夹形式内含exe入口执行文件的形式,也可以是一个单独的exe文件。0 P% Y5 E. B2 G- |
    9 }$ Y1 O9 w7 S
    fbs:基于PyInstaller,使用起来更加方便
    . @& g$ R. ]- p2 ^9 t$ ^
    ; ^# j6 R* o7 u- a" Z这里选择了fbs来打包。fbs的安装方法:! {) n5 g8 }5 k

    . w) H6 m) `( k3 K) X0 O1 gpip install fbs
    9 |; P0 I: L9 J" E+ L0 T3 A使用方法,在命令行中输入:
    & T# N3 [, u. M4 H5 R8 S  {+ p" N* x& ~7 h0 h
    fbs startproject
    3 N+ {% x+ h' t9 E7 P5 `执行完成后需要输入一些APP的名称等。完成后会生成如下目录:
    , [  l0 m% u$ e6 T6 O$ m( ~% @; }0 u" r6 S+ D

    7 z, A% O& c; n
    + F! `7 X4 a0 [; X将刚才编写的PyQt5的代码(demo.py和Weather.py)拖到src/main/python文件夹下,删除原有的main.py,并将demo.py修改为main.py。然后打开 main.py,在文件头部添加如下代码:
    9 C  l& `" c3 N) W0 v) \8 A/ U2 \3 S0 s) ~( [1 K3 |2 w6 v
    from fbs_runtime.application_context.PyQt5 import ApplicationContext
    4 Y+ r$ B# X6 E```
    # n4 C- U: z6 W) c" P  G5 T  T完成后执行:, M& F  ~, F$ a" i; K
    ```: u& d% V0 [( A
    fbs freeze0 o' j3 Q4 x" J! H* [- p
    ```
    6 R+ ]" ~# h& `) V& d0 q0 g7 V即可实现打包。生成的exe可执行文件在\target\MyApp文件下。- |! J' s( P* @/ N
    * _$ y5 I- G, ~2 I! I5 Y6 s
    ————————————————  @& ]% Q9 P4 p" X( b
    版权声明:本文为CSDN博主「宋宋讲编程」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。5 V) h$ b  F6 n  z0 T- @' D
    原文链接:https://blog.csdn.net/qiqi1220/article/details/126289667
    # S: y3 {: N5 {) n
    3 |6 q8 V; ~4 W3 C: ?5 ]
    6 U- F, v% N- o# B6 M1 t& l
    # n/ q1 U, F' k0 _
    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-15 10:55 , Processed in 0.440633 second(s), 51 queries .

    回顶部