0

0

Python 多线程 Socket 服务器正确启动与主线程并发执行教程

霞舞

霞舞

发布时间:2026-01-17 21:16:02

|

732人浏览过

|

来源于php中文网

原创

Python 多线程 Socket 服务器正确启动与主线程并发执行教程

本文详解如何修复 python socket 服务器中因线程阻塞导致主线程无法继续执行的问题,重点解决 `thread.start()` 后主线程“卡死”的常见误区,并提供可稳定运行的多客户端服务实现方案。

在开发基于 socket 和 threading 的服务器程序时,一个典型误区是:误以为调用 thread.start() 就能自动释放主线程控制权,而忽略了被启动线程内部逻辑(如无限 while True 循环)是否真正“非阻塞”或“可退出”,以及主线程自身是否被意外阻塞。

你遇到的问题——运行 test.py 后只看到 "Server is running and listening ..." 反复输出,却看不到 "A" 或 "." 打印——根本原因并非线程未启动,而是 server.run() 方法本身在主线程中被阻塞式调用,且其内部 self.server.accept() 是同步阻塞操作。但更关键的是:你在 test.py 中虽然用 thread.start() 启动了 server.run,却遗漏了 thread.daemon = False(默认即非守护线程)+ thread.start() 后未做任何主线程让步或调度保障,而实际问题出在另一处:server.run() 内部的 accept() 调用虽阻塞,但线程已正确启动;真正导致 test.py 主循环不执行的原因,是 print("A") 所在的 while True 循环缺乏显式延时,造成 CPU 占用过高、输出缓冲未刷新、甚至在某些环境下被调度器压制。

✅ 正确做法需同时满足三点:

  • ✅ server.run() 必须在独立线程中启动(你已做到);
  • stream() 方法必须接收 client 套接字作为参数(避免共享状态竞争),且不能依赖类属性(如 self.client)——否则多客户端时会相互覆盖;
  • ✅ 主线程循环必须包含 time.sleep() + flush=True,防止 I/O 缓冲和高频率空转干扰调度。

以下是重构后的生产就绪代码:

立即学习Python免费学习笔记(深入)”;

server.py

Mistral AI
Mistral AI

Mistral AI被称为“欧洲版的OpenAI”,也是目前欧洲最强的 LLM 大模型平台

下载
import socket
import threading
import time

class Server:
    def __init__(self, host: str, port: int):
        self.host = host
        self.port = port
        self.server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        # 允许端口重用,避免 TIME_WAIT 错误
        self.server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        self.server.bind((host, port))
        self.server.listen(5)  # 设置合理连接队列长度
        print(f"Server listening on {host}:{port}")

    def stream(self, client: socket.socket, address):
        """为每个客户端分配独立线程,发送心跳消息"""
        print(f"[+] New connection from {address}")
        try:
            while True:
                client.send(b'PING\n')
                time.sleep(2)
        except (BrokenPipeError, ConnectionResetError, OSError):
            print(f"[-] Client {address} disconnected")
        finally:
            client.close()

    def run(self):
        """主监听循环 —— 每接受一个连接,就启一个新线程处理"""
        try:
            while True:
                client, address = self.server.accept()
                # 关键:将 client 和 address 作为参数传入 target 函数
                thread = threading.Thread(
                    target=self.stream,
                    args=(client, address),
                    daemon=True  # 设为守护线程,避免主程序退出时残留
                )
                thread.start()
        except KeyboardInterrupt:
            print("\n[!] Server shutting down...")
        finally:
            self.server.close()

test.py

from server import Server
import threading
import time

if __name__ == "__main__":
    # 绑定到 '' 表示监听所有接口(含局域网),便于树莓派连接
    server = Server('', 8001)

    # 在后台线程启动服务器
    server_thread = threading.Thread(target=server.run, name="ServerThread")
    server_thread.daemon = False
    server_thread.start()

    # 主线程持续运行,打印状态点(带 flush 防止缓冲)
    print("Main thread active. Press Ctrl+C to exit.")
    try:
        while True:
            print(".", end="", flush=True)
            time.sleep(1)
    except KeyboardInterrupt:
        print("\n[!] Exiting main thread...")

? 关键注意事项:

  • self.server.accept() 是阻塞调用,但它发生在 server.run() 所在线程内,不会阻塞 test.py 的主线程——只要 server.run() 确实跑在独立线程中(本例已确保);
  • 若未加 time.sleep(),主线程 while True: print("A") 会以最大频率刷屏,可能因输出缓冲未及时刷新而看似“无输出”,或触发系统级调度抑制;
  • stream() 中使用 b'PING\n' 替代空字符串或 self.message,避免因 self.message 为空导致 send() 发送零字节(合法但无意义),且多客户端时共享 self.message 会导致数据污染;
  • 设置 daemon=True 在 stream() 线程中,确保主程序退出时自动清理子线程,防止僵尸连接;
  • 实际部署建议添加异常捕获(如 OSError 处理 accept() 中断)、日志记录及连接数限制。

运行后,你将看到:

  • test.py 每秒输出一个 .,证明主线程正常运行;
  • 终端同时打印 Server is running... 和 Connection established...,表明服务线程与主线程完全解耦;
  • 使用 nc 192.168.178.30 8001 或 Python socket.connect() 即可收到连续 PING 响应。

至此,你的跨设备键盘指令传输服务器已具备稳定、可扩展、易调试的多线程基础架构。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

758

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

639

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

761

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

618

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1265

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

548

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

579

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

708

2023.08.11

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 2.9万人学习

Django 教程
Django 教程

共28课时 | 3.2万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号