1.进程
data:image/s3,"s3://crabby-images/babaa/babaab4ea860924f3c6c745c3551c8c14616d44c" alt=""
1.1多任务的引入
单任务
data:image/s3,"s3://crabby-images/cf3e8/cf3e823ec0d99d1b02f8728cd29e63ef1ae7aac5" alt=""
1.2多任务
data:image/s3,"s3://crabby-images/158fc/158fccdf0fbaa7897d472920983ba3a53f7fb25d" alt=""
说明:
·程序执行到os.fork()时,操作系统会创建一个新的进程(子进程),然后复制父进程的所有信息到子进程
·然后父进程和子进程都会从fork()函数中得到一个返回值,在子进程中这个值一定是0,而父进程中是子进程的id号。
注意:
子进程永远返回 0,而父进程返回 子进程的 I D
data:image/s3,"s3://crabby-images/521d1/521d1f5d4dcf756378a91437365766ff8fce3573" alt=""
2.多次fork问题
data:image/s3,"s3://crabby-images/af42a/af42a5cbb0a5336102abbf7bc1309be0591369d1" alt=""
这时候会产生四个进程。
data:image/s3,"s3://crabby-images/bf73a/bf73ad3fcf94f6dd9600b264c7866f2be809b99f" alt=""
多进程修改全局变量:
data:image/s3,"s3://crabby-images/d5b91/d5b91814ca3b01e778e92d103f0f0ee55f9ba885" alt=""
打印的结果: 哈哈1---- num=1 哈哈2---- num=1
总结:
多进程中,每个进程中所有数据(包括全局变量)都各有拥有一份,互 不影响。也就是说相当于有两个num。互不干扰。
3. getpid():获得这个进程的id
解释:哪个进程运行了这个代码,就获得这个进程编号的pid
getppid():获得这个进程的父进程的id
解释:哪个进程运行了这个代码,就获得这个进程的父进程编号的pid
代码:
import os
rpid = os.fork()
if rpid<0:
print("fork调用失败。")
elif rpid ==0:
print("我是子进程(%s),我的父进程是(%s)"%(os.getpid(),os.getppid()))
else:
print("我是父进程(%s),我的子进程是(%s)"%(os.getpid(),rpid))
print("父子进程都可以执行这里的代码")
运行结果:
我是父进程(19360),我的子进程是(19361)
父子进程都可以执行这里的代码
我是子进程(19361),我的父进程是(19360)
父子进程都可以执行这里的代码。
4.multiprocessing
Python是跨平台的,也提供一个跨平台的多进程。multiprocessing模块就是跨平台版本的多进程模块
multiprocessing模块提供了一个Process类来代表一个进程对象。
Process语法结构如下:
Process([group [, target [, name [, args [, kwargs]]]]])
·target:表示这个进程实例所调用对象;
·args:表示调用对象的位置参数元组;
·kwargs:表示调用对象的关键字参数字典;
·name:为当前进程实例的别名;
·group:大多数情况下用不到;
Process类常用方法:
·is_alive():判断进程实例是否还在执行;
·join([timeout]):是否等待进程实例执行结束,或等待多少秒;
·start():启动进程实例(创建子进程);
·run():如果没有给定target参数,对这个对象调用start()方法时,就将执行对象中的run()方法;
·terminate():不管任务是否完成,立即终止;
Process类常用属性:
·name:当前进程实例别名,默认为Process-N,N为从1开始递增的整数;
·pid:当前进程实例的PID值;
data:image/s3,"s3://crabby-images/16bd0/16bd04630950c178816aaa2987d59ba7dfc08e60" alt=""
data:image/s3,"s3://crabby-images/76037/76037e5d845ef28f57ac1a427128ca9a1380740a" alt=""
data:image/s3,"s3://crabby-images/bfeb6/bfeb65d44ff0e231a4db14d96c97e7cd2c0b4a2c" alt=""
join()的作用:
data:image/s3,"s3://crabby-images/44f95/44f95153ad2d09ced0b694ff7c03242be15999ba" alt=""
data:image/s3,"s3://crabby-images/79c68/79c6892e1f0da78a89b873b286d26f09801adbd7" alt=""
join(timeout)的作用:
data:image/s3,"s3://crabby-images/2d43c/2d43cfc93b7c6910377c2a71a0c58f92a2ee79ac" alt=""
data:image/s3,"s3://crabby-images/58981/589817d3d21e84ffb086993393c4f8be69bc33ac" alt=""
设置守护进程:
data:image/s3,"s3://crabby-images/3e20c/3e20c1a96c920a915039c8963773307b2bb89101" alt=""
data:image/s3,"s3://crabby-images/c370f/c370f13de5d12afc58e3380c59f31a34adcf653b" alt=""
多进程 唱歌跳舞例子:
data:image/s3,"s3://crabby-images/f8942/f894206d8cf176cc87732db23cd6c205d1f744e3" alt=""
data:image/s3,"s3://crabby-images/4d7e2/4d7e25b02bf61bad0cd7f0a6392be8c69808b4cd" alt=""
两个子进程.join()的运行顺序:
data:image/s3,"s3://crabby-images/f704b/f704b4d7a87240803522629079791a3377b96f13" alt=""
data:image/s3,"s3://crabby-images/5ac6b/5ac6b694729b6402e3d860a84e130508444b6751" alt=""
两个子进程,一个设置成守护进程,对另一个不受影响。:
data:image/s3,"s3://crabby-images/fa6e0/fa6e070e29eb427232c0941e734e3cdd886eb694" alt=""
data:image/s3,"s3://crabby-images/591b2/591b26d4c3d1664afb37834076ddcde44625de45" alt=""
第二种实现多进程的方法:创建-Process子类
data:image/s3,"s3://crabby-images/08529/08529a87ceb47c00a0e8f9114686fd99596677e0" alt=""
data:image/s3,"s3://crabby-images/6b6aa/6b6aa2ba18c70476c2fe8108156ac5d3d5dc149f" alt=""
继承类创建的多进程的唱歌跳舞例子:
data:image/s3,"s3://crabby-images/26435/2643503bf94500f0cb04e99c739ac1309203afac" alt=""
data:image/s3,"s3://crabby-images/12741/12741e2e0a3122e6fcccf3f60ac3179f8abb112c" alt=""
data:image/s3,"s3://crabby-images/6d665/6d66588b4cb13c832e7e0b8f62356d60967266c8" alt=""
两种方式的对比:(创建多进程有两种方式)
1、方法
2、继承类
继承类是以面向对象考虑这个事的,所以业务逻辑复杂,建议使用继承类,更好理解
5.进程池Pool
data:image/s3,"s3://crabby-images/34179/341791d9750eab9988906c1c84b865b0e1b0fab2" alt=""
apply_async非堵塞: 谁先运行完下一个接着上去。
data:image/s3,"s3://crabby-images/392dc/392dc45e95439bf731b38335d518b85c99b19d23" alt=""
data:image/s3,"s3://crabby-images/b792e/b792ebbbcf384768f8150643ca71b8e661479eae" alt=""
apply堵塞: 一个运行完在运行下一个。
data:image/s3,"s3://crabby-images/c67bc/c67bcfe8bcfc57285f4e2fd9cec4ad709026514b" alt=""
data:image/s3,"s3://crabby-images/4e02c/4e02cfd8366280b40b4e1166f42ec2b92113a004" alt=""
6.进程间通信-Queue
Process之间有时需要通信,操作系统提供了很多机制来实现进程间的通信
可以使用multiprocessing模块的Queue实现多进程之间的数据传递
说明:
初始化Queue() 对象时(例如:q=Queue()),若括号中没有指定最大可接收的消息数量,或数量为负值,那么就代表可接受的消息数量没有上限(直到内存的尽头);
·Queue.qsize():返回当前队列包含的消息数量;
·Queue.empty():如果队列为空,返回True,反之False;
·Queue.full():如果队列满了,返回True,反之False;
·Queue.get([block[, timeout]]):获取队列中的一条消息,然后将其从列队中移除,block默认值为True; 这里面可以加参数。
·Queue.get_nowait():相当Queue.get(False);
·Queue.put(item,[block[, timeout]]):将item消息写入队列,block默认值为True;
Queue.put_nowait(item):相当Queue.put(item, False)
例1:
data:image/s3,"s3://crabby-images/2ee9f/2ee9f71472d6e729cdeb93f8a43680b5d712d8b1" alt=""
data:image/s3,"s3://crabby-images/10df9/10df9994cebabc60cd2b9815ea9b3d105ab9a8a0" alt=""
例2:
data:image/s3,"s3://crabby-images/e33da/e33da4bcb40299e3158c48f5861886349e83a500" alt=""
data:image/s3,"s3://crabby-images/99de3/99de36540dbae97e081c5c3752498177890e0974" alt=""
例子3:利用apply阻塞写完再读
data:image/s3,"s3://crabby-images/5614c/5614c45b1979909fe2f928c08713a4a9b2ceded1" alt=""
data:image/s3,"s3://crabby-images/d620a/d620a25ea151b8b0e161ff69c1961b235aa151ae" alt=""
例4:apply_async异步
data:image/s3,"s3://crabby-images/69c95/69c95b5c3c74013ff3ea8ee6cce707010d32827a" alt=""
data:image/s3,"s3://crabby-images/ec6e3/ec6e3109aa5f6c5dcb7e9542a1337f9f7382f027" alt=""
网友评论