网站首页 > 教程文章 正文
1.软件打包安装程序
利用setupfactory软件进行客户端软件打包,形成一个setup安装程序。
2. 建立FTP服务器
利用quickeasyftpserver在远程服务器中建立FTP服务器,放置2个文件,一个setup安装文件,一个是ver.txt文件,里面写上软件的版本号12位,例如20200917V001。
3. 客户端程序开发
在客户端程序中点击升级按钮后
(1) 程序首先删除本地的上一次的版本和升级文件,然后从ftp服务器上下载ver.txt,然后判断服务器的setup版本和当前软版本是否一致,不一致,就提示用户升级。
(2) 如果用户同意升级,就在ftp上下载setup安装文件。
(3) 下载完成后,再开一个线程执行setup程序,然后把当前的程序关闭,安装程序会覆盖当前的程序。
问题:
这种方法比较简单,主要问题是不知道ftp文件的下载进度,会有一直卡死的现象。这个原因是进度条的定时器程序刷新UI和FTP下载程序在一个线程中,造成拥塞了。下载时就不能定时中断,刷新UI了
解决方案:
(1) 对于ftp下载采用异步方式,下载完后发送信号给主线程。
(2) 主线程点击升级按钮,绑定子线程和对应的回调函数。然后启动ftp子线程。
(3) 在回调函数中,执行此程序就是下载结束了,这时可以在主线程中启动,进度条满格,然后执行分线程执行安装程序,并在主程序中把自己结束
代码如下:
from update import Ui_MainWindow
from PyQt5 import QtWidgets
from PyQt5.QtWidgets import QMessageBox
import sys
import os
from ftplib import FTP # 引入ftp模块
import win32process
from PyQt5.QtCore import QBasicTimer, QThread, pyqtSignal
import time
mycurrent_ver = '20200916V001' # 当前软件版本号
mynew_ver = ''
#异步线程
class MyCal(QThread):
#自定义一个信号名
cal_signal = pyqtSignal(int) #定义信号返回的数值类型
#构造函数
def __init__(self, mypath, filename,parent=None): #
super(MyCal, self).__init__(parent)
self.mypath = mypath
self.filename=filename
print("分线程:",self.mypath,self.filename)
#析构函数
def __del__(self):
self.wait()
#该线程主程序
def run(self):
##ftp下载程序版本程序
ftp = MyFtp('127.0.0.1)
ftp.login('admin', '123456')
ftp.downloadFile(self.mypath, '/down/', self.filename)
ftp.close()
print("分线程下载结束:", self.mypath+self.filename)
self.cal_signal.emit(1) #发射信号,传参数
class MyFtp:
ftp = FTP()
def __init__(self, host, port=21):
self.ftp.connect(host, port)
def login(self, username, pwd):
self.ftp.set_debuglevel(2) # 打开调试级别2,显示详细信息
self.ftp.login(username, pwd)
print(self.ftp.welcome)
def downloadFile(self, localpath, remotepath, filename):
os.chdir(localpath) # 切换工作路径到下载目录
self.ftp.cwd(remotepath) # 要登录的ftp目录
self.ftp.nlst() # 获取目录下的文件
file_handle = open(filename, "wb").write # 以写模式在本地打开文件
self.ftp.retrbinary('RETR %s' % os.path.basename(filename), file_handle, blocksize=1024) # 下载ftp文件
# ftp.delete(filename) # 删除ftp服务器上的文件
def close(self):
self.ftp.set_debuglevel(0) # 关闭调试
self.ftp.quit()
# 主窗体
class MainForm(QtWidgets.QMainWindow, Ui_MainWindow):
def __init__(self):
super(MainForm, self).__init__()
self.setupUi(self)
self.timer = QBasicTimer()
self.step = 0
# self.myupdate_bt()
# 进度条控制定时器
def timerEvent(self, event):
if self.step >= 100:
self.timer.stop()
self.pushButton_2.setText('开始进度条')
self.step = 0
self.progressBar.setValue(self.step)
return
self.step = self.step + 1
if self.step>=99:
self.step=99
self.progressBar.setValue(self.step)
# 进度条控制
def myprocessbar_bt(self):
pass
if self.timer.isActive():
self.timer.stop()
self.pushButton_2.setText('开始进度条')
else:
self.timer.start(100, self)
self.pushButton_2.setText('停止进度条')
def myupdate_bt(self):
pass
myhomedir = os.getcwd()
mypath = os.getcwd() + '\\down\\'
myfilepath1 = mypath + "ver.txt"
myfilepath2 = mypath + "setup.exe"
# 删除文件
x1 = os.path.exists(myfilepath1) # True/False
if x1 == True:
print("删除文件:" + myfilepath1)
os.remove(myfilepath1)
x1 = os.path.exists(myfilepath2) # True/False
if x1 == True:
print("删除文件:" + myfilepath2)
os.remove(myfilepath2)
##ftp下载程序版本程序
ftp = MyFtp('127.0.0.1)
ftp.login('admin', '123456')
ftp.downloadFile(mypath, '/down/', 'ver.txt')
ftp.close()
print("下载结束:", myfilepath1)
# 判断版本
with open(myfilepath1, 'rt') as f1:
mynew_ver = f1.readline()[0:12]
self.label_4.setText(mynew_ver)
self.label_2.setText(mycurrent_ver)
if mynew_ver != mycurrent_ver:
x1 = QMessageBox.information(self, "确认信息", "有新版本,确认是否升级", QMessageBox.Yes | QMessageBox.No)
if x1 == QMessageBox.No:
os.chdir(myhomedir)
return
print("开始升级")
QMessageBox.information(self, "确认信息", "开始下载程序!!")
self.myprocessbar_bt()#显示进度条
# self.timer.start(100, self)
##拥塞ftp下载程序
# ftp = MyFtp('127.0.0.1')
# ftp.login('admin', '123456')
# ftp.downloadFile(mypath, '/down/', 'setup.exe')
# ftp.close()
# print("下载结束:", myfilepath2)
# QMessageBox.information(self, "确认信息", "下载程序结束!!")
####异步FTP下载
self.cal = MyCal(mypath, 'setup.exe') # 点击按钮后新建计算的线程
self.cal.cal_signal.connect(self.cal_callback) # 连接计算线程的信号
self.cal.start() # 开始运行线程
else:
QMessageBox.information(self, "确认信息", "已经是最新版本,不用升级")
os.chdir(myhomedir)
return
# 省略 使用接收到线程返回的参数
###执行第三方程序,新线程
#win32process.CreateProcess(myfilepath2, '', None, None, 0, win32process.CREATE_NO_WINDOW, None, None,win32process.STARTUPINFO())
os.chdir(myhomedir)
print("主程序退出")
#sys.exit()
def cal_callback(self): # 接收到计算线程信号后的回掉函数
pass
print('异步ftp下载结束')
self.step=100
self.progressBar.setValue(self.step)
#self.myprocessbar_bt()#显示进度条
myhomedir = os.getcwd()
mypath = myhomedir
myfilepath1 = mypath + "\\ver.txt"
myfilepath2 = mypath + "\\setup.exe"
QMessageBox.information(self, "确认信息", "下载结束,开始升级")
print(myfilepath2)
win32process.CreateProcess(myfilepath2, '', None, None, 0, win32process.CREATE_NO_WINDOW, None, None,win32process.STARTUPINFO())
sys.exit()
# 主程序入口
if __name__ == "__main__":
app = QtWidgets.QApplication(sys.argv)
win_main = MainForm()
win_main.show()
sys.exit(app.exec_())
界面如下
- 上一篇: 大神总结的Qt开发经验,满满的都是干货
- 下一篇: QT小技巧 qt的教程
猜你喜欢
- 2024-12-17 Qt使用教程:创建Qt Quick UI表单(三)
- 2024-12-17 一小例子,了解 TCP 通讯流程 | Qt 示例
- 2024-12-17 如何使用QT编写自己的串口调试助手软件
- 2024-12-17 Qt pro文件中的常用宏说明 qt pro宏定义
- 2024-12-17 Qt Creator 源码学习笔记01,初识QTC
- 2024-12-17 QT学习:statusBar的使用,创建控件添加到statusBar上面
- 2024-12-17 一口气,了解 Qt 的所有 IPC 方式 | Qt 速学
- 2024-12-17 Qt线程QThread开启和安全退出 qt线程怎么安全关闭
- 2024-12-17 Qt之使用socket实现远程控制 qt使用mqtt
- 2024-12-17 C/C++编程笔记:编写完成了一个C/C++程序,如何做一个界面出来?
- 最近发表
- 标签列表
-
- location.href (44)
- document.ready (36)
- git checkout -b (34)
- 跃点数 (35)
- 阿里云镜像地址 (33)
- qt qmessagebox (36)
- md5 sha1 (32)
- mybatis plus page (35)
- semaphore 使用详解 (32)
- update from 语句 (32)
- vue @scroll (38)
- 堆栈区别 (33)
- 在线子域名爆破 (32)
- 什么是容器 (33)
- sha1 md5 (33)
- navicat导出数据 (34)
- 阿里云acp考试 (33)
- 阿里云 nacos (34)
- redhat官网下载镜像 (36)
- srs服务器 (33)
- pico开发者 (33)
- https的端口号 (34)
- vscode更改主题 (35)
- 阿里云资源池 (34)
- os.path.join (33)