当前位置: 首页 > news >正文

Python+QT远程控制助手-ver2

程序示例精选
Python+QT远程控制助手
如需安装运行环境或远程调试,见文章底部个人QQ名片,由专业技术人员远程协助!

前言

这篇博客针对《Python+QT远程控制助手》编写代码,代码整洁,规则,易读。 学习与应用推荐首选。


文章目录

一、所需工具软件
二、使用步骤
       1. 主要代码
       2. 运行结果
三、在线协助

一、所需工具软件

       1. Python
       2. Pycharm

二、使用步骤

代码如下(示例):
def hash_password(password):return hashlib.sha256((password + PASSWORD_SALT).encode('utf-8')).hexdigest()
def send_message(sock, message):data = json.dumps(message).encode('utf-8')sock.sendall(len(data).to_bytes(4, 'big') + data)
def recv_message(sock):numT2 = 0length_bytes = sock.recv(4)if not length_bytes:return Nonelength = int.from_bytes(length_bytes, 'big')data = b''while len(data) < length:chunk = sock.recv(min(4096, length - len(data)))if not chunk:return Nonedata += chunknumT2 += 1print("recv_message : ", numT2)return json.loads(data.decode('utf-8'))
class RemoteControlClient(QWidget):def __init__(self):super().__init__()def send_keepalive(self):while True:time.sleep(30)if self.sock:try:send_message(self.sock, {'type': 'keepalive','timestamp': int(time.time())})# print("主控端发送心跳包")except Exception as e:print("主控端心跳包发送失败:", e)breakdef connect_remote(self):self.target_id = self.id_input.text().strip()password = self.pwd_input.text().strip()if not self.target_id or not password:QMessageBox.warning(self, "错误", "请输入目标ID和密码")returnself.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)try:self.sock.connect((SERVER_HOST, SERVER_PORT))self.sock.settimeout(300)  # 这里加上except Exception as e:QMessageBox.warning(self, "错误", f"连接服务器失败: {e}")return# 注册(主控端用随机ID即可)#my_id = "CTRL" + str(int(time.time()))[-6:]my_id = "CTRL_MAIN"send_message(self.sock, {'type': 'register','data': {'device_id': my_id, 'password_hash': hash_password(password)},'timestamp': int(time.time())})# 等待注册确认while True:msg = recv_message(self.sock)if msg and msg.get('type') == 'register_ack' and msg.get('status') == 'success':break#注册成功后启动心跳包threading.Thread(target=self.send_keepalive, daemon=True).start()# 发起连接请求send_message(self.sock, {'type': 'peer_message','target_device_id': self.target_id,'data': json.dumps({'type': 'CONNECT_REQUEST','source_device_id': my_id,'password': hash_password(password),'timestamp': int(time.time())}),'timestamp': int(time.time())})self.label.setText("等待对方接受...")threading.Thread(target=self.recv_loop, daemon=True).start()def recv_loop(self):print("recv_loop")while True:try:msg = recv_message(self.sock)if not msg:print("连接断开,准备重连...")self.auto_reconnect()breakif msg.get('type') == 'peer_message':data = msg.get('data')+ data.get('reason', ''))self.connected = Falseelif dtype == 'SCREEN_DATA':self.show_screen(data)elif dtype == 'SCREEN_DIFF':self.show_screen_diff(data)print("recv_loop clsoe")except socket.timeout:print("接收超时,准备重连...")self.auto_reconnect()breakexcept Exception as e:print("recv error:", e)self.auto_reconnect()breakdef auto_reconnect(self):self.label.setText("断线,正在重连...")try:if self.sock:self.sock.close()except:passtime.sleep(3)  # 等待5秒再重连self.connect_remote()def request_screen(self):if self.connected and self.sock:send_message(self.sock, {'type': 'peer_message','target_device_id': self.target_id,'data': json.dumps({'type': 'REQUEST_SCREEN'}),'timestamp': int(time.time())})def show_screen(self, data):try:img_data = base64.b64decode(data['data'])img_bytes = BytesIO(img_data)from PIL import Imagepil_img = Image.open(img_bytes)w, h = pil_img.sizeself.last_screen_size = (w, h)pil_img = pil_img.convert("RGB")arr = np.array(pil_img)h, w, ch = arr.shapebytes_per_line = ch * wqimg = QImage(arr.data, w, h, bytes_per_line, QImage.Format_RGB888)pix = QPixmap.fromImage(qimg)self.label.setPixmap(pix.scaled(self.label.size(), Qt.KeepAspectRatio, Qt.SmoothTransformation))self.label.setScaledContents(True)# 差异重组用self.full_img = arr.copy()except Exception as e:print("show_screen error:", e)def show_screen_diff(self, data):try:w, h = data.get('width'), data.get('height')block_size = data.get('block_size', 256)if self.full_img is None or self.full_img.shape[0] != h or self.full_img.shape[1] != w:self.full_img = np.zeros((h, w, 3), dtype=np.uint8)updated = Falsefor blk in data['blocks']:bx, by = blk['x'], blk['y']bw, bh = blk['w'], blk['h']img_data = base64.b64decode(blk['data'])from PIL import Imagepil_img = Image.open(BytesIO(img_data)).convert("RGB")arr = np.array(pil_img)self.full_img[by:by + bh, bx:bx + bw, :] = arrupdated = Trueif updated:self.last_screen_size = (w, h)arr = self.full_imgh, w, ch = arr.shapebytes_per_line = ch * wqimg = QImage(arr.data, w, h, bytes_per_line, QImage.Format_RGB888)pix = QPixmap.fromImage(qimg)self.label.setPixmap(pix.scaled(self.label.size(), Qt.KeepAspectRatio, Qt.SmoothTransformation))self.showT+=1print("showT : ",self.showT)except Exception as e:print("show_screen_diff error:", e)
运行结果

在这里插入图片描述

三、在线协助:

如需安装运行环境或远程调试,见文章底部个人 QQ 名片,由专业技术人员远程协助!

1)远程安装运行环境,代码调试
2)Visual Studio, Qt, C++, Python编程语言入门指导
3)界面美化
4)软件制作
5)云服务器申请
6)网站制作

当前文章连接:https://blog.csdn.net/alicema1111/article/details/132666851
个人博客主页:https://blog.csdn.net/alicema1111?type=blog
博主所有文章点这里:https://blog.csdn.net/alicema1111?type=blog

博主推荐:
Python人脸识别考勤打卡系统:
https://blog.csdn.net/alicema1111/article/details/133434445
Python果树水果识别:https://blog.csdn.net/alicema1111/article/details/130862842
Python+Yolov8+Deepsort入口人流量统计:https://blog.csdn.net/alicema1111/article/details/130454430
Python+Qt人脸识别门禁管理系统:https://blog.csdn.net/alicema1111/article/details/130353433
Python+Qt指纹录入识别考勤系统:https://blog.csdn.net/alicema1111/article/details/129338432
Python Yolov5火焰烟雾识别源码分享:https://blog.csdn.net/alicema1111/article/details/128420453
Python+Yolov8路面桥梁墙体裂缝识别:https://blog.csdn.net/alicema1111/article/details/133434445
Python+Yolov5道路障碍物识别:https://blog.csdn.net/alicema1111/article/details/129589741
Python+Yolov5跌倒检测 摔倒检测 人物目标行为 人体特征识别:https://blog.csdn.net/alicema1111/article/details/129272048

http://www.dtcms.com/a/250847.html

相关文章:

  • Dify 插件工具:远程连接配置指南
  • NY271NY274美光科技固态NY278NY284
  • 车载通信架构 --- IP ECU 在连接被拒绝后的重连机制
  • C++第一阶段——语言基础与核心特性
  • 项目 : 基于正倒排的boost搜索引擎
  • AI for Science:智能科技如何重塑科学研究
  • 基于机器学习的逐巷充填开采岩层运动地表沉降预测
  • 408第一季 - 数据结构 - B树与B+树
  • Spring Boot Web 应用开发
  • 6,TCP客户端
  • 【技术实战】工业级设备健康管理系统搭建全栈指南:从数据采集到预测性维护
  • LeetCode 2300.咒语和药水的成功对数
  • Java面试题:分布式ID时钟回拨怎么处理?序列号耗尽了怎么办?
  • 香橙派Zero3结合Docker部署私有音乐实践过程
  • vscode通过ssh连接
  • Flink与Kubernetes集成
  • RV1126+OPENCV对视频流单独进行视频膨胀/腐蚀操作
  • 微软Bing正式推出AI视频生成工具:Bing Video Creator,由Sora技术驱动,限时免费体验!
  • Docker + PyFlink1.17 数据写入 MySQL
  • 05-mcp-server案例分享-用豆包大模型 1.6 手搓文生图视频 MCP-server发布到PyPI官网
  • ESP32 005 MicroPython I2S 实现音频传输与播放
  • 性能优化 - 案例篇:11种优化接口性能的通用方案
  • Bootstrap 5学习教程,从入门到精通, Bootstrap 5 列表组(List Group)语法知识点及案例(14)
  • AWS EC2 终极指南:如何选择预装 GPU 驱动和特定功能的最佳 AMI
  • AWS 解决方案深度剖析:Amazon QLDB — 构建可信赖、不可变的数据审计基石
  • 基于AWS无服务器架构的区块链API集成:零基础设施运维实践
  • 微软azure抢跑aws和谷歌云的区别
  • 线性代数(2)几何角度来理解线性方程组
  • Java面试题022:一文深入了解微服务网关Gateway
  • 【微服务】134:SpringCloud