当前位置: 首页 > 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.xdnf.cn/news/1053631.html

相关文章:

  • 《注解的江湖:一场元数据的“宫斗剧”》
  • 每日算法刷题Day32 6.15:leetcode枚举技巧7道题,用时1h10min
  • 计网复习知识(17)应用层
  • jQuery 3D透明蓄水池状柱状图插件
  • IDA动态调试环境配置全流程
  • 【Markdown】基础用法汇总(标题、列表、链接、图片、加粗斜体、上下角标、引用块、代码块、公式)
  • 学习日记-day30-6.15
  • Linux安装LLaMA Factory
  • Netty 全面深入学习指南
  • 项目实训个人工作梳理
  • 【算法 day03】LeetCode 203.移除链表元素 | 707.设计链表 | 206.反转链表
  • nodejs中Express框架的基本使用
  • ​​信息系统项目管理师-项目范围管理 知识点总结与例题分析​​
  • Claude Code 实用教程——使用方法详解
  • 庙算兵棋推演AI开发初探(8-神经网络模型接智能体进行游戏)
  • 文本预测和分类任务
  • [笔记] 基于esp32s3用GUI-Guider-1.9.1-GA开发LVGL界面
  • 认识电子元器件之磁传感器
  • Spring有代理对象的循环依赖时,如何确保代理对象能够正确持有原始对象的所有属性赋值结果?
  • 234. 回文链表
  • SQL 增删改查 —— 笔记篇
  • 面向对象设计原则
  • 深度学习——基于卷积神经网络实现食物图像分类【3】(保存最优模型)
  • React19源码系列之Hooks(useState)
  • Linux中的连接符
  • 谐波减速器 MINIF8 和 MINIF11 的区别
  • 事务传播机制分析:用户注册场景分析
  • 日语学习-日语知识点小记-进阶-JLPT-真题训练-N2阶段(2):2020年12月2018年7月
  • leetcode148-排序链表
  • 《Java编程思想》读书笔记:第十二章