如何在类内部封装多线程逻辑并运行多个对象实例
技术百科
聖光之護
发布时间:2026-01-20
浏览: 次 
本文介绍如何将双线程执行逻辑封装进 python 类中,使每个对象实例自主管理其内部的两个并发线程,避免在主程序中手动创建和管理大量线程对象。
在面向对象编程中,良好的封装性意味着对象应尽可能自治——包括其并发行为。原始写法将 threading.Thread 的创建、启动与等待(join)全部暴露在类外部,不仅破坏了封装,还导致主逻辑臃肿、可维护性差,且难以扩展(例如增加线程数或统一生命周期管理)。
通过将线程管理内聚到类中,我们可以定义清晰的接口:run_threads() 启动本实例的全部工作线程,join_threads() 等待它们完成。这样,每个 MyObject 实例都成为一个“可并发执行的单元”。
以下是优化后的完整实现:
import threading
import time
class MyObject:
def __init__(self, item_id):
self.item_id = item_id
def func_one(self):
print(f"[{self.item_id}] func_one started")
time.sleep(1.5) # 模拟耗时操作
print(f"[{self.item_id}] func_one finished")
def func_two(self):
print(f"[{self.item_id}] func_two started")
time.sleep(1.0)
print(f"[{self.item_id}] func_two finished")
def run_threads(self):
"""启动本实例的两个工作线程"""
self.thread1 = threading.Thread(target=self.func_one, name=f"{self.item_id}-func1")
self.thread2 = threading.Thread(target=self.func_two, name=f"{self.item_id}-func2")
self.thread1.start()
self.thread2.start()
def join_threads(self):
"""阻塞等待本实例所有工作线程结束"""
self.thread1.join()
self.thread2.join()
# 使用示例:创建并并发运行多个实例
obj1 = MyObject("item-01")
obj2 = MyObject("item-02")
# 启动所有实例的内部线程(非阻塞)
obj1.run_threads()
obj2.run_threads()
# 等待全部完成(顺序调用,但各实例内线程仍并行执行)
obj1.join_threads()
obj2.join_threads()
print("All instances completed.")✅ 关键优势:
- 高内聚:线程生命周期与对象绑定,self.thread1/self.thread2 属于实例状态;
- 易复用:新增实例只需 objN = MyObject(...); objN.run_threads(); objN.join_threads();
- 可扩展:如需支持动态线程数,可将 func_one/func_two 抽象为列表或注册函数;
- 便于调试:可通过 threading.current_thread().name 或日志明确归属。
⚠️ 注意事项:
- 避免在 __init__ 中直接启动线程(可能导致对象未完全构造就执行);
- 若需线程安全共享状态,请使用 threading.Lock 或其他同步原语;
- join_threads() 必须在 run_threads() 之后调用,否则会引发 RuntimeError(尝试 join 未启动的线程);
- 如需异步等待或超时控制,可改用 thread.join(timeout=...) 并检查 thread.is_alive()。
这种设计既符合 OOP 原则,又保持了多线程的灵活性,是构建可伸缩并发组件的推荐实践。
# 成为一个
# 多个
# python
# 我们可以
# 只需
# 如需
# 或其他
# 并发
# 对象
# 主程序
# 接口
# 线程
# 异步
# 多线程
# 封装
# 类中
# Thread
# 装进
# 封装性
# 面向对象
# 请使用
# 线程生命周期
# 面向对象编程
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- 如何在 Go 中判断变量是否为函数类型
- Windows怎样拦截QQ浏览器广告_Window
- 当网站SEO排名下降时,如何应对?
- 如何使用Golang捕获并记录协程panic_保证
- Windows怎样关闭开始菜单推荐广告_Windo
- 如何使用Golang安装API文档生成工具_快速生
- c++怎么操作redis数据库_c++ hired
- C++如何使用std::transform批量处理
- Python包结构设计_大型项目组织解析【指导】
- mac怎么看硬盘大小_MAC查看磁盘存储空间与文件
- Win11怎么开启游戏模式_Windows11优化
- Windows10系统怎么查看系统版本_Win10
- php485读数据时阻塞怎么办_php485非阻塞
- Win11摄像头无法使用怎么办_Win11相机隐私
- Win11怎么开启空间音效_Windows11耳机
- mac怎么右键_MAC鼠标右键设置与触控板手势技巧
- Flask 表单数据通过 SMTP 发送邮件的完整
- Win11怎么关闭SmartScreen_禁用Wi
- PHP 中如何在函数内持久化修改引用变量的指向
- VSC怎么在PHP中调试MySQL_数据库交互排查
- Win11怎么关闭专注助手 Win11关闭免打扰模
- Python 模块的 __name__ 属性如何由
- Windows11如何设置专注助手_Windows
- Python对象生命周期管理_创建销毁解析【教程】
- Mac上的iMovie如何剪辑视频?(新手入门教程
- C++友元类使用场景_C++类间协作设计方式讲解
- php转mp4怎么设置帧率_调整php生成mp4视
- Win10如何卸载微软拼音输入法 Win10只保留
- 如何使用Golang实现微服务状态监控_Golan
- Windows10电脑怎么设置虚拟光驱_Win10
- Go 中实现 Python urllib.quot
- Windows10系统服务优化指南_Win10禁用
- Win10如何更改开机密码_Windows10登录
- Win11如何设置电源计划_Win11电源计划优化
- php8.4如何配置ssl证书_php8.4htt
- c++怎么编写动态链接库dll_c++ __dec
- 如何在 Go 中创建包含 map 的 slice(
- 如何使用Golang sync.Map实现并发安全
- PowerShell怎么创建复杂的XML结构
- mac怎么安装字体_MAC添加第三方字体与字体册管
- php订单日志怎么按状态筛选_php筛选不同状态订
- Windows电脑如何截屏?(四种快捷方法)
- 如何在Golang中实现自定义Benchmark_
- Win11麦克风没声音怎么设置_Win11麦克风权
- c++怎么设置线程优先级与cpu亲和性_c++ 多
- Win11怎么开启自动HDR画质_Windows1
- php删除数据怎么清空表_truncate与del
- Python并发安全问题_资源竞争说明【指导】
- windows如何备份注册表_windows导出和
- mac怎么查看wifi密码_MAC查看已连接WiF

QQ客服