(进程和线程)[https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0014319272686365ec7ceaeca33428c914edf8f70cca383000]
多任务的实现有3种方式:
多进程模式;
多线程模式;
多进程+多线程模式。
Python既支持多进程,又支持多线程
小结:
线程是最小的执行单元,而进程由至少一个线程组成。如何调度进程和线程,
完全由操作系统决定,程序自己不能决定什么时候执行,执行多长时间
(多进程)[https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/001431927781401bb47ccf187b24c3b955157bb12c5882d000#0]
Unix/Linux操作系统提供了一个fork()系统调用
普通的函数调用,调用一次,返回一次,但是fork()调用一次,返回两次,
因为操作系统自动把当前进程(称为父进程)复制了一份(称为子进程),然后,
分别在父进程和子进程内返回
1 | import os #window执行失败 window没有fork函数 |
输出Process (876) start…
I (876) just created a child process (877).
I am child process (877) and my parent is 876.
multiprocessing
multiprocessing模块就是跨平台版本的多进程模块
1 | from multiprocessing import Process #导入跨平台模块 |
Parent process 928.
Process will start.
Run child process test (929)…
Process end.
join()方法可以等待子进程结束后再继续往下运行,通常用于进程间的同步
pool 进程池
如果要启动大量的子进程,可以用进程池的方式批量创建子进程
1 | from multiprocessing import Pool |
执行结果:
Parent process 669.
Waiting for all subprocesses done…
Run task 0 (671)…
Run task 1 (672)…
Run task 2 (673)…
Run task 3 (674)…
Task 2 runs 0.14 seconds.
Run task 4 (673)…
Task 1 runs 0.27 seconds.
Task 3 runs 0.86 seconds.
Task 0 runs 1.41 seconds.
Task 4 runs 1.91 seconds.
All subprocesses done.
代码解读
对Pool对象调用join()方法会等待所有子进程执行完毕,调用join()之前必须先调用close(),
调用close()之后就不能继续添加新的Process了
请注意输出的结果,task 0,1,2,3是立刻执行的,而task 4要等待前面某个task完成后才执行,
这是因为Pool的默认大小在我的电脑上是4,因此,最多同时执行4个进程。这是Pool有意设计的限制,
并不是操作系统的限制。如果改成:
p = Pool(5)
就可以同时跑5个进程。
由于Pool的默认大小是CPU的核数,如果你不幸拥有8核CPU,你要提交至少9个子进程才能看到上面的等待效果。
子进程
subprocess模块可以让我们非常方便地启动一个子进程,然后控制其输入和输出
1 | import subprocess |
相当于命令行运行 nslookup www.python.org1
2
3$ nslookup www.python.org
Server: 192.168.19.4
Address: 192.168.19.4#53
如果子进程还需要输入,则可以通过communicate()方法输入:1
2
3
4
5
6
7import subprocess
print('$ nslookup')
p = subprocess.Popen(['nslookup'], stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
output, err = p.communicate(b'set q=mx\npython.org\nexit\n')
print(output.decode('utf-8'))
print('Exit code:', p.returncode)
上面的代码相当于在命令行执行命令nslookup,然后手动输入:
set q=mx
python.org
exit
进程间通信
Process之间肯定是需要通信的,操作系统提供了很多机制来实现进程间的通信。Python的multiprocessing模块包装了底层的机制,提供了Queue、Pipes等多种方式来交换数据。
我们以Queue为例,在父进程中创建两个子进程,一个往Queue里写数据,一个从Queue里读数据
1 | from multiprocessing import Process, Queue |
执行结果1
2
3
4
5
6
7
8Process to write: 50563
Put A to queue...
Process to read: 50564
Get A from queue.
Put B to queue...
Get B from queue.
Put C to queue...
Get C from queue.
小结
在Unix/Linux下,可以使用fork()调用实现多进程。
要实现跨平台的多进程,可以使用multiprocessing模块。
进程间通信是通过Queue、Pipes等实现的。
- (进程和线程)[https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/0014319272686365ec7ceaeca33428c914edf8f70cca383000]
- (多进程)[https://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/001431927781401bb47ccf187b24c3b955157bb12c5882d000#0]
- multiprocessing
- pool 进程池