Python——pyqt5的计算器(源码+打包)

目录

一、效果图

二、源码

三、如何打包

四、如何减小打包程序大小(方法1)

五、如何减小打包程序大小(方法2)

学习视频


一、效果图

只是单纯的练手,然后再学习一下如何打包

 Python——pyqt5的计算器(源码+打包)_第1张图片

二、源码

calculator_UI.zip - 蓝奏云

三、如何打包

安装PyInstaller

pip install pyinstaller

快速打包命令:

pyinstaller -wF -i logo.ico main.py

main.py:是我要打包的python项目文件

 -i:为main.exe文件指定的图标(如果没有,则不需要该参数)

logo.ico :生成.exe文件的图标(推荐一个在线转换ico图标的网址)

-F:生成结果是一个exe文件,所有的第三方依赖、资源和代码均被打包进该exe内

-w :不显示命令行窗口(如不用此参数,打包后的程序在运行时会弹出一个命令行窗口)


打包成功

 Python——pyqt5的计算器(源码+打包)_第2张图片

  1. 基本用法 :  pyinstaller [选项] 文件名.py

  2. 常用选项

  3. -F, --onefile:将应用程序打包为单个可执行文件

  4. -D, --onedir:将应用程序打包为目录

  5. -c, --console:使用控制台模式(在终端窗口中运行应用程序)

  6. -w, --noconsole:使用Windows模式(没有控制台窗口)

  7. --name:指定应用程序的名称

  8. --distpath:指定输出目录

  9. --clean:在打包之前清除临时文件夹日志

  10. -d, --debug:生成运行时调试器

  11. --log-level:设置日志级别,供调试嵌入其他文件

  12. --add-data:将data文件夹嵌入到生成的可执行文件中

  13. --add-binary:将binary文件与其他文件一起嵌入到可执行文件中

  14. --hidden-import:打包特定的Python库,例如--hidden-import=PIL

  15. --version:显示版本信息

  16. --help:显示帮助信息

以上是_pyinstaller的常用选项和命令,可以根据需要进行选择和使用。

有时候,打包exe程序后的文件大小会非常大!如何减小呢?

四、如何减小打包程序大小(方法1)

1、安装pipenv

pip install pipenv

2、新建一个文件夹

安装完成后,随便找一个盘符,在这个盘符的根目录(最好是根目录)新建一个文件夹,作为虚拟环境,然后双击进入该文件夹

3、安装python版本(在虚拟环境下安装所需的python版本)

pipenv install -python 3.6
  • 系统有可能提示无法转换为 “utf-8”,具体原因是因为中文转码的问题,但是不必深究。
  • 这时候最好的方法是重启电脑
  • 重启电脑后,不要打开任何其他程序,重复上述第2步

4、激活虚拟环境(在命令行下)

pipenv shell
  • 如何系统提示无法转换为“utf-8”,则再次重启电脑
  • 然后重新进入power shell,再尝试运行 pipenv shell

5、安装需要的库(仅需安装需要打包的.py文件中所用到的库)

根据要打包的程序中需要导入的库,在pipenv环境下重新安装,例如:

pipenv install pyinstaller
pipenv install removebg

6、进行打包

把py脚本文件复制到这个新建的目录下,重新运行 pyinstaller

pyinstaller -wF -i logo.ico main.py

 五、如何减小打包程序大小(方法2)

使用 PyInstaller 将 PyQt5 程序打包成 exe 文件,并用 UPX 压缩生成的文件,即可生成尽可能小的 exe 文件。

1、安装 PyInstaller 和 UPX

打开命令行窗口(Windows 下可以按下 Win+R 后,输入 cmd 启动),输入以下命令安装 PyInstaller 和 UPX:

pip install pyinstaller pip install upx

2、创建 PyInstaller 打包配置文件

在 PyInstaller 命令行中输入以下命令,创建一个打包配置文件 myapp.spec

pyinstaller -y -w myapp.py

-y 表示自动覆盖现有的 build 和 dist 文件夹

-w 表示去除控制台窗口

此时当前目录会出现一个 build 文件夹和一个 dist 文件夹,dist 文件夹中会生成一个不压缩的 exe 文件。

3、修改 PyInstaller 打包配置文件

在当前目录下,使用编辑器打开 myapp.spec 文件,添加以下代码:

import upx
upx_path = "your_upx_path" # UPX 所在路径

a = Analysis(['myapp.py'],
             pathex=['your_path_for_py_file'], # myapp.py 所在路径
             binaries=[],
             datas=[],
             hiddenimports=['sip', 'PyQt5.QtCore', 'PyQt5.QtGui', 'PyQt5.QtWidgets'],
             hookspath=['hooks'],
             runtime_tmpdir=None,
             excludes=[],
             win_no_prefer_redirects=False,
             win_private_assemblies=False,
             cipher=None,
             alg='AES256',
             key=None,
             upx=True, # 是否开启压缩
             upx_path=upx_path, # UPX 所在路径
             console=False)
pyz = PYZ(a.pure, a.zipped_data,
             cipher=None,
             alg='AES256',
             key=None,
             unbuffered=False)
exe = EXE(pyz,
          a.scripts,
          a.binaries,
          a.zipfiles,
          a.datas,
          [],
          name='myapp',
          debug=False,
          bootloader_ignore_signals=False,
          strip=False,
          upx=True, # 经过测试,tostring() 或 tobytes() 方法生成的资源文件即是这里的 a.zipped_data
          upx_path=upx_path, # UPX 所在路径
          upx_exclude=[],
          runtime_tmpdir=None,
          console=False,
          icon=None)

注意:

upx_path 需要填写 UPX.exe 所在的文件夹路径

hiddenimports 需要添加 PyQt5 相关的模块

4、打包 exe 文件

在命令行中输入以下命令,将 myapp.spec 文件转换为 exe 文件:

pyinstaller -y -c myapp.spec

生成的文件在 dist 文件夹内,文件名是在 myapp.spec 文件中 name 参数指定的。

-c 表示使用控制台程序,如果不需要控制台程序,可以将 -c 改为 -w

5、使用 UPX 压缩 exe 文件

在命令行中输入以下命令,使用 UPX 压缩 exe 文件:

upx --best myapp.exe

myapp.exe 是需要压缩的文件名

--best 表示使用最高级别的压缩


附代码

import sys

from PyQt5 import QtCore, QtWidgets
from PyQt5.QtCore import Qt
from PyQt5.QtGui import QColor
from PyQt5.QtWidgets import QMainWindow, QApplication

from calculator import Ui_calculator_MainWindow



class MyMainForm(QMainWindow, Ui_calculator_MainWindow):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.setupUi(self)
        self.start_x = None
        self.start_y = None
        self.anim=None
        self.setAttribute(QtCore.Qt.WA_TranslucentBackground)
        self.setWindowFlags(Qt.FramelessWindowHint)  # 设置窗口标志:隐藏窗口边框


        # Bind button scale up and scale down
        self.btn_max.clicked.connect(self.btn_max_click)
        self.btn_min.clicked.connect(self.minimize)
        self.btn_stop.clicked.connect(self.closeEvent)

        # 输入
        self.input = ''

        # 绑定方法
        self.bin_number()

    # 输入显示
    def add_input(self,input):
        self.input +=input
        self.number_input.setPlainText(str(self.input))

    # 计算结果
    def calculate_res(self):
        try:
            self.number_output.display(eval(self.input))
        except Exception:
            self.number_input.setPlainText('输入格式错误!')
            self.input = ''
        finally:
            # self.input = ''
            pass

    # clear LCD number
    def clear_input(self):
        self.input = ''
        self.number_input.setPlainText('')

    # back one number
    def back_number(self):
        self.input = self.input[:-1]
        self.number_input.setPlainText(self.input)

    # add button events
    def bin_number(self):
        # number events
        self.b_one.clicked.connect(lambda: self.add_input('1'))
        self.b_tow.clicked.connect(lambda: self.add_input('2'))
        self.b_three.clicked.connect(lambda: self.add_input('3'))
        self.b_four.clicked.connect(lambda: self.add_input('4'))
        self.b_five.clicked.connect(lambda: self.add_input('5'))
        self.b_six.clicked.connect(lambda: self.add_input('6'))
        self.b_seven.clicked.connect(lambda: self.add_input('7'))
        self.b_eight.clicked.connect(lambda: self.add_input('8'))
        self.b_nine.clicked.connect(lambda: self.add_input('9'))

        self.b_point.clicked.connect(lambda: self.add_input('.'))

        self.b_add.clicked.connect(lambda: self.add_input('+'))
        self.b_divide.clicked.connect(lambda: self.add_input('/'))
        self.b_multiple.clicked.connect(lambda: self.add_input('*'))
        self.a_substract.clicked.connect(lambda: self.add_input('-'))

        # calculate result
        self.b_equal.clicked.connect(lambda: self.calculate_res())

        # back one number
        self.b_back.clicked.connect(lambda: self.back_number())
        # clear input
        self.b_clear.clicked.connect(lambda: self.clear_input())


    # Window scale up and scale down
    def btn_max_click(self):
        if self.isMaximized():
            self.showNormal()
            # self.btn_stop.setText('放大窗口')
        else:
            self.showMaximized()
            # self.btn_stop.setText('缩小窗口')

    # 重写 closeEvent 函数,关闭窗口并停止程序运行
    def closeEvent(self, event):
        sys.exit()

    # 最小化窗口
    def minimize(self):
        self.showMinimized()

    # 鼠标左键释放时被触发
    def mouseReleaseEvent(self, event):
        self.start_x = None
        self.start_y = None

    # 在鼠标左键按下时被触发
    def mousePressEvent(self, event):
        if event.button() == QtCore.Qt.LeftButton:
            super(MyMainForm, self).mousePressEvent(event)
            self.start_x = event.x()
            self.start_y = event.y()

    # 在鼠标移动时被触发
    def mouseMoveEvent(self, event):
        try:
            super(MyMainForm, self).mouseMoveEvent(event)
            dis_x = event.x() - self.start_x
            dis_y = event.y() - self.start_y
            self.move(self.x() + dis_x, self.y() + dis_y)
        except:
            pass

    # 窗口设置阴影效果
    def effect_shadow_style(self, widget):
        effect_shadow = QtWidgets.QGraphicsDropShadowEffect(self)
        effect_shadow.setOffset(12, 12)  # 偏移
        effect_shadow.setBlurRadius(128)  # 阴影半径
        effect_shadow.setColor(QColor(155, 230, 237, 150))  # 阴影颜色
        widget.setGraphicsEffect(effect_shadow)



    # 背景渐变
    # def paintEvent(self, event):
    #     painter = QPainter(self)
    #     gradient = QLinearGradient(0, 0, self.width(), self.height())
    #     gradient.setColorAt(0, QColor(89, 217, 212, 128))
    #     gradient.setColorAt(1, QColor(104, 202, 237, 128))
    #     gradient.setStart(0, 0)
    #     gradient.setFinalStop(self.width(), self.height())
    #     painter.setBrush(gradient)
    #     painter.drawRect(self.rect())




if __name__ == "__main__":


    app = QApplication(sys.argv)
    myWin = MyMainForm()
    myWin.show()


    # myWin.btn_img.clicked.connect(click())


    sys.exit(app.exec_())

学习视频

【已完结】PySide6百炼成真,带你系统性入门Qt_哔哩哔哩_bilibili【已完结】PySide6百炼成真,带你系统性入门Qt共计75条视频,包括:000 新的课程介绍、002环境搭建、003基础框架等,UP主更多精彩视频,请关注UP账号。https://www.bilibili.com/video/BV1c84y1N7iL计算器这个案例就是跟着这个视频里做的!打call!很赞的教程!

你可能感兴趣的:(Python,windows)