python并发编程之semaphore(信号量)_浅谈Python并发编程之进程(守护进程、锁、信号量)...

前言:本博文是对Python并发编程之进程的知识延伸,主要讲解:守护进程、锁、信号量。

友情链接:

一、守护进程(daemon)

1.1 守护进程概念

首先我们都知道:正常情况下,主进程默认等待子进程调用结束之后再结束,守护进程在主进程所有代码执行完毕之后,自动终止。当然我们也可以使用kill -9 进程号,杀死进程。

那么接下来,我们看一下守护进程的语法:

进程对象.daemon = True

通过上面的代码,我们就可以设置该进程为守护进程。

注意: 必须要写在start()方法之前赋值。

此时,将设置为主进程守护,主进程如果代码执行结束了,该守护进程自动结束。

1.2 守护进程示例

from multiprocessing import Process

import time

def func():

print("子进程start")

time.sleep(1)

print("子进程end")

if __name__ == "__main__":

p = Process(target=func)

p.daemon = True

p.start()

time.sleep(1)

print("主进程执行结束")

因为为守护主进程,主进程结束子进程立即结束。

1.3 多个子进程的情况

当多个子进程并发执行时,默认主进程等待子进程,如果标记该子进程是守护进程,当主进程执行完毕所有代码之后,守护进程立刻终止。

主进程的代码执行到最后一行,就意味着函数代码执行完毕,此时就应该杀掉守护进程。其他非守护进程继续正常执行,主进程仍然等待直到结束,最后主进程在真正的释放结束。

from multiprocessing import Process

import time

def func1():

count = 1

while True:

print("*" * count)

time.sleep(0.5)

count +=1

def func2():

print("func2 start")

time.sleep(3)

print("func2 end")

if __name__ == "__main__":

p1 = Process(target=func1)

p1.daemon = True

p1.start()

p2 = Process(target=func2)

p2.start()

print("主进程代码执行结束...")

二、互斥锁(Lock)

上锁:lock.acquire()

解锁:lock.release()

2.1 互斥锁概念

同一时间允许一个进程上一把锁,就是Lock。那么我们加锁的意义在哪呢?加锁可以保证多个进程修改同一块数据时,同一时间只能有一个任务可以进行修改,即串行的修改。虽然我们的程序运行速度是慢了,但牺牲速度却保证了数据安全。

同一时间允许多个进程上多把锁 就是[信号量Semaphore]。

信号量是锁的变形:实际实现是 计数器 + 锁,同时允许多个进程上锁。

2.2 互斥锁作用

互斥锁Lock:互斥锁就是进程的相互排斥。

谁先抢到自由,谁就上锁该资源内容,这样做可以保证数据的同步性。

注意:多个锁一起上,不开锁,会造成死锁,上锁和解锁是一对。

2.3 互斥锁示例

from multiprocessing import Process, Lock

# 创建一把锁

lock = Lock()

# 上锁

lock.acquire()

print(1)

# 解锁

lock.release()

# 死锁 : 只上锁,不解锁,会阻塞,产生死锁

lock.acquire()

print(2)

lock.release()

print(3)

lock.acquire()

print(4)

#lock.release():在这里我们不解锁

lock.acquire()

print(5)

注意:输出结果没有5,因为上一个锁,没有解锁,造成了死锁。

2.4 区分同步和异步

在产生进程对象的时候,进程之间是异步,上锁之后,进程之间变成同步。

from multiprocessing import Process,Lock

def func(num, lock):

lock.acquire()

print("走到上锁这个地方,变成一个同步程序,先来的进行先执行,后来的进程后执行,按次序依次执行")

print(num)

lock.release()

if __name__ == "__main__":

# lock互斥锁, 进程之间数据不共享

# 但是lock对象底层是通过socket来互相发送数据,不管多少进程,都是同一个lock锁

lock = Lock()

for i in range(3):

p = Process(target=func, args=(i, lock))

# 1. 10个子进程异步执行,是并发操作

p.start()

三、Semaphore(信号量)

信号量Semaphore是一个计数器,控制对公共资源或者临界区域的访问量,信号量可以指定同时访问资源或者进入临界区域的进程数。每次有一个进程获得信号量时,计数器-1,若计数器为0时,其他进程就停止访问信号量,一直阻塞直到其他进程释放信号量。

我们之前说到的Lock,属于互斥锁,也就是一把钥匙配备一把锁,同时只允许锁住某一个数据。而信号量则是多把钥匙配备多把锁,也就是说同时允许锁住多个数据。

from multiprocessing import Process,Semaphore

import random,time

def ktv(person,sem):

# 上锁

sem.acquire()

print("%s进入ktv唱歌" % (person))

time.sleep(random.randrange(3, 8))

print("%s走出ktv离开" % (person))

# 解锁

sem.release()

if __name__ == "__main__":

# 同一时间最多允许2个进程执行ktv任务,剩下的进程等待

sem = Semaphore(2)

for i in range(1, 6):

p = Process(target=ktv,args=("person%s" %i, sem))

p.start()

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值