猪行天下之Python基础——9.1 Python多线程与多进程(上)(一)

简介: 内容简述: 线程与进程的相关概念  1、程序,进程,线程,多进程,多线程 2、线程的生命周期 3、并行与并发,同步与异步 4、线程同步安全 5、与锁有关的特殊情况:死锁,饥饿与活锁 6、守护线程 7、线程并发的经典问题:生产中与消费者问题 8、Python中的GIL锁 9、Python中对多线程与多进程的支持

线程与进程的相关概念


关于线程和进程的话题,大部分的书只是微微提下,读者学完云里雾里,不知所以。本章会对Python中的多线程和多进程进行详解。大部分都是概念性的东西,不要去死记硬背,学完了解有个大概印象就好。


1、程序,进程,线程,多进程,多线程


关于程序,进程和线程的一些名词概念如图所示:



有句非常经典的话:“进程是资源分配的最小单位,线程则是CPU调度的最小单位”。

先说说「多进程」:从普通用户的视角:


如果你的电脑是Windows的话,Ctrl+Alt+Del打开任务管理器,可以看到电脑运行着很多的进程,比如QQ,微信,网易云音乐等。这就是多进程,每个进程各司其职完成对应的功能,互不干扰,你聊天的时候音乐照常播放。


再说说开发仔的视角:


多进程的概念更倾向于:多个进程协同地区完成同一项工作。


问题:为什么要在应用里使用多进程


笔者观点:摆脱系统的一些限制和为自己的应用获取更多的资源,举个例子:

在Android系统中会为每个应用(进程)限制最大内容,单个进程超过这个阈值会引起OOM,而使用多进程技术可以规避这个内存溢出的问题。再举个例子:Python在实现Python解析器(CPython)时引入了GIL锁,使得任何时候仅有一个线程在执行,多线程的效率可能还比不上单线程,使用多进程技术可以规避这个限制。


再说说「多线程」,首先为什么会引入线程的概念呢?举个例子:


你有一个文本程序,三个功能组成部分:接收用户的输入,显示到屏幕上,保存到硬盘里,如果由三个进程组成:输入接收进程A,显示内容进程B,写入硬盘进程C,而他们之间共同需要拥有的东西——文本内容,而进程A,B,C运行在不同的内存空间,这就涉及到进程通信问题了,而频繁的进程切换势必导致性能上的损失。有没有一种机制使得做这三个任务时共享资源呢?这个时候线程(轻量级的进程)就派上用场了,多个线程共享进程数据。相信读者看到这里,对于多进程和多线程的概念应该有个初步的了解了,接下来简单比较下两者的优劣和使用场景:


对比内容 多进程 多线程
内存与CPU 内存占用多,切换复杂,
CPU利用率低
内存占用少,切换简单,
CPU利用率高
数据共享与同步 数据共享复杂,需要通过IPC,
因为数据是分开的,同步简单
共享进程数据,数据共享简单,但也因为
这个原因导致同步复杂
创建销毁和切换 创建销毁和切换复杂,速度慢 创建销毁和切换简单,速度很快
编程和调试 编程简单,调试简单 编程复杂,调试复杂
可靠性 进程间不会互相影响 一个线程挂掉将导致整个进程挂掉
分布式 适应于多核、多机分布式;如果一台
机器不够,扩展到多台机器比较简单
适应于多核分布式


2、线程的生命周期


线程的生命周期如图所示



各个状态说明


  • New(新建),新创建的线程进过初始化,进入Runnable(就绪)状态。


  • Runnable(就绪),等待线程调度,调度后进入Running(运行)状态。


  • Running(运行),线程正常运行,期间可能会因为某些情况进入Blocked
    (同步锁;调用了sleep()和join()方法进入Sleeping状态;执行wait()方法进入
    Waiting状态,等待其他线程notify通知唤醒)。


  • Blocked(堵塞),线程暂停运行,解除堵塞后进入Runnable(就绪)状态重新等待调度。


  • Dead(死亡):线程完成了它的任务正常结束或因异常导致终止。


3、并行与并发,同步与异步


并行与并发的区别:


并行是同时处理多个任务,而并发则是处理多个任务,而不一定要同时,并行可以说是并发的子集


同步与异步的区别:


  • 同步:线程执行某个请求,如果该请求需要一段时间才能返回信息,那么这个线程会一直等待,直到收到返回信息才能继续执行下去。


  • 异步:线程执行完某个请求,不需要一直等直接继续执行后续操作,当有消息
    返回时系统会通知线程进程处理,这样可以提高执行的效率;异步在网络请求
    的应用非常常见。


目录
打赏
0
0
0
0
7
分享
相关文章
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
在计算机系统的底层架构中,操作系统肩负着资源管理与任务调度的重任。当我们启动各类应用程序时,其背后复杂的运作机制便悄然展开。程序,作为静态的指令集合,如何在系统中实现动态执行?本文带你一探究竟!
【Linux进程概念】—— 操作系统中的“生命体”,计算机里的“多线程”
Python GIL(全局解释器锁)机制对多线程性能影响的深度分析
在Python开发中,GIL(全局解释器锁)一直备受关注。本文基于CPython解释器,探讨GIL的技术本质及其对程序性能的影响。GIL确保同一时刻只有一个线程执行代码,以保护内存管理的安全性,但也限制了多线程并行计算的效率。文章分析了GIL的必要性、局限性,并介绍了多进程、异步编程等替代方案。尽管Python 3.13计划移除GIL,但该特性至少要到2028年才会默认禁用,因此理解GIL仍至关重要。
157 16
Python GIL(全局解释器锁)机制对多线程性能影响的深度分析
|
23天前
|
python3多线程中使用线程睡眠
本文详细介绍了Python3多线程编程中使用线程睡眠的基本方法和应用场景。通过 `time.sleep()`函数,可以使线程暂停执行一段指定的时间,从而控制线程的执行节奏。通过实际示例演示了如何在多线程中使用线程睡眠来实现计数器和下载器功能。希望本文能帮助您更好地理解和应用Python多线程编程,提高程序的并发能力和执行效率。
47 20
Python实用技巧:轻松驾驭多线程与多进程,加速任务执行
在Python编程中,多线程和多进程是提升程序效率的关键工具。多线程适用于I/O密集型任务,如文件读写、网络请求;多进程则适合CPU密集型任务,如科学计算、图像处理。本文详细介绍这两种并发编程方式的基本用法及应用场景,并通过实例代码展示如何使用threading、multiprocessing模块及线程池、进程池来优化程序性能。结合实际案例,帮助读者掌握并发编程技巧,提高程序执行速度和资源利用率。
19 0
|
5天前
|
Linux 进程前台后台切换与作业控制
进程前台/后台切换及作业控制简介: 在 Shell 中,启动的程序默认为前台进程,会占用终端直到执行完毕。例如,执行 `./shella.sh` 时,终端会被占用。为避免不便,可将命令放到后台运行,如 `./shella.sh &`,此时终端命令行立即返回,可继续输入其他命令。 常用作业控制命令: - `fg %1`:将后台作业切换到前台。 - `Ctrl + Z`:暂停前台作业并放到后台。 - `bg %1`:让暂停的后台作业继续执行。 - `kill %1`:终止后台作业。 优先级调整:
29 5
探索Linux中的`ps`命令:进程监控与分析的利器
探索Linux中的`ps`命令:进程监控与分析的利器
172 13
掌握taskset:优化你的Linux进程,提升系统性能
在多核处理器成为现代计算标准的今天,运维人员和性能调优人员面临着如何有效利用这些处理能力的挑战。优化进程运行的位置不仅可以提高性能,还能更好地管理和分配系统资源。 其中,taskset命令是一个强大的工具,它允许管理员将进程绑定到特定的CPU核心,减少上下文切换的开销,从而提升整体效率。
掌握taskset:优化你的Linux进程,提升系统性能
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
227 4
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
探索进程调度:Linux内核中的完全公平调度器
【8月更文挑战第2天】在操作系统的心脏——内核中,进程调度算法扮演着至关重要的角色。本文将深入探讨Linux内核中的完全公平调度器(Completely Fair Scheduler, CFS),一个旨在提供公平时间分配给所有进程的调度器。我们将通过代码示例,理解CFS如何管理运行队列、选择下一个运行进程以及如何对实时负载进行响应。文章将揭示CFS的设计哲学,并展示其如何在现代多任务计算环境中实现高效的资源分配。
【Linux】冯诺依曼体系结构与操作系统及其进程
【Linux】冯诺依曼体系结构与操作系统及其进程
211 1
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等