python基础——锁

news2025/1/13 15:47:52

进程锁 (互斥锁)

进程锁的引入:

模拟抢票程序:

from multiprocessing import Process
import json
import time
def show_ticket(i):
    with open("./tickets.txt",mode="r",encoding="utf-8") as file:
        ticket = json.load(file)
    print("{}:当前的票数是:{}".format(i,ticket['count']))
def buy_ticket(i):
    with open("./tickets.txt",mode="r",encoding="utf-8") as file:
        ticket = json.load(file)
        if ticket['count'] > 0:
            ticket['count'] -= 1
            print("{}买到票了".format(i))
        else:
            print("票卖完了")
        time.sleep(0.1)
        with open("./tickets.txt",mode="w",encoding="utf-8") as new_file:
            json.dump(ticket,new_file)

if __name__ == '__main__':
    for i in range(10):
        Process(target=show_ticket,args=(i,)).start()
        Process(target=buy_ticket,args=(i,)).start()

运行结果:

实际仅有一张票,但是由于系统的并发执行速度较快,导致系统出现错误

此时,就需要使用锁来进行数据保护,防止出现数据上的错误

进程锁的使用:

from multiprocessing import Process,Lock
import json
import time
def show_ticket(i,lock):
    lock.acquire()
    with open("./tickets.txt",mode="r",encoding="utf-8") as file:
        ticket = json.load(file)
    print("{}:当前的票数是:{}".format(i,ticket['count']))
    lock.release()
def buy_ticket(i,lock):
    lock.acquire()
    with open("./tickets.txt",mode="r",encoding="utf-8") as file:
        ticket = json.load(file)
        if ticket['count'] > 0:
            ticket['count'] -= 1
            print("{}买到票了".format(i))
        else:
            print("票卖完了")
        time.sleep(0.1)
        with open("./tickets.txt",mode="w",encoding="utf-8") as new_file:
            json.dump(ticket,new_file)
        lock.release()
if __name__ == '__main__':
    lock = Lock()
    for i in range(10):
        Process(target=show_ticket,args=(i,lock)).start()
        Process(target=buy_ticket,args=(i,lock)).start()

运行结果:

使用进程锁可以保证一次进运行一个进程,防止进程之间数据的错误

进程锁再次使用:

不使用进程锁:

from multiprocessing import Process
import time
from multiprocessing import Lock
def func1(i,lock):
    # lock.acquire()
    print("函数第{}次执行".format(i))
    time.sleep(1)
    print("函数执行完毕")
    # lock.release()
if __name__ == '__main__':
    lock = Lock()
    for i in range(10):
        Process(target=func1,args=(i,lock)).start()

执行结果:

使用进程锁:

from multiprocessing import Process
import time
from multiprocessing import Lock
def func1(i,lock):
    lock.acquire()
    print("函数第{}次执行".format(i))
    time.sleep(1)
    print("函数执行完毕")

if __name__ == '__main__':
    lock = Lock()
    for i in range(10):
        Process(target=func1,args=(i,lock)).start()

执行结果:

注意:由于前者未使用进程锁,因此十个进程并发执行,总执行时间1s

后者使用进程锁,需要一个进程一个进程进行执行,因此总执行时间为10s

部分代码解释:

lock = Lock()   创建一个锁对象

lock.acquire()   表示该进程拿走锁,然后执行,阻塞其他进程

lock.release()  表示该进程执行完毕,归还锁,使得其他进程得以继续执行

此时,若果进程中只有lock.acquire()方法,而没有lock.release()方法,会使得程序阻塞,无法继续向下进行

from multiprocessing import Process
import time
from multiprocessing import Lock
def func1(i,lock):
    lock.acquire()
    print("函数第{}次执行".format(i))
    time.sleep(1)
    print("函数执行完毕")

if __name__ == '__main__':
    lock = Lock()
    for i in range(10):
        Process(target=func1,args=(i,lock)).start()

执行结果:

另一种写法:

with lock:

代替lock.acquire()方法与lock.release()方法

线程锁              

线程锁的引入:

from threading import Thread
n = 0
def add():
    for i in range(2200000):
        global n
        n += 1
def sub():
    for i in range(2200000):
        global n
        n -= 1

p1 = Thread(target=add)
p1.start()
p2 = Thread(target=sub)
p2.start()
p1.join()
p2.join()
print(n)

运行结果不确定,由于线程的执行速度过快,在同一时间,可能会有多个线程在进行加操作,而在加操作完成后,赋值操作仅进行了一次,因此出现数值的错误,因为错误次数不确定,因此运行的结果也不确定

部分运行结果:

线程中锁的使用:

from threading import Thread,Lock
n = 0
def add(lock):
    for i in range(2200000):
        global n
        lock.acquire()
        n += 1
        lock.release()
def sub(lock):
    for i in range(2200000):
        global n
        lock.acquire()
        n -= 1
        lock.release()
if __name__ == '__main__':
    lock = Lock()
    p1 = Thread(target=add,args=(lock,))
    p1.start()
    p2 = Thread(target=sub,args=(lock,))
    p2.start()
    p1.join()
    p2.join()
    print(n)

此时,无论程序执行多少次,数值的大小如何,由于锁的使用,每次的加运算与赋值运算都是单次进行,因此,数值不会出现错误

执行结果:

递归锁

在一个线程中可以进行多次acquire()操作

但同时,只有进行多次release()操作,才能解锁其他进程

递归锁的使用:

from threading import Thread,Lock,RLock
import time
n = 0
def add(lock):
    for i in range(2):
        global n
        lock.acquire()
        lock.acquire()
        n += 1
        time.sleep(1)
        print("函数add的执行次数{}".format(n))
        lock.release()
def sub(lock):
    for i in range(2):
        global n
        lock.acquire()
        n -= 1
        print("函数sub的执行次数{}".format(n))
        lock.release()
if __name__ == '__main__':
    lock = RLock()
    p1 = Thread(target=add,args=(lock,))
    p1.start()
    p2 = Thread(target=sub,args=(lock,))
    p2.start()
    p1.join()
    p2.join()
    print(n)

执行结果:

此时可以看到,由于函数add中上锁了两次,而只进行了一次解锁,因此在函数add执行完毕之后函数sub并不会进行执行

死锁现象

由于进程的执行顺序不当,或者资源的分配不当而导致整个进程不能进行执行,而陷入卡死的状态

死锁示例:

rom threading import Thread,Lock
import time
noodle_lock = Lock()
fork_lock = Lock()

def eat(name):
    noodle_lock.acquire()
    print("{}抢到面了".format(name))
    fork_lock.acquire()
    print("{}抢到叉子了".format(name))
    print("{}吃面".format(name))
    time.sleep(0.1)
    fork_lock.release()
    print("{}放下了叉子".format(name))
    noodle_lock.release()
    print("{}放下面了".format(name))
def eat2(name):
    fork_lock.acquire()
    print("{}抢到了叉子".format(name))
    noodle_lock.acquire()
    print("{}抢到面了".format(name))
    print("{}吃面".format(name))
    time.sleep(0.1)
    noodle_lock.release()
    print("{}放下面了".format(name))
    fork_lock.release()
    print("{}放下叉子了".format(name))

Thread(target=eat,args=(1,)).start()
Thread(target=eat2,args=(2,)).start()
Thread(target=eat,args=(3,)).start()
Thread(target=eat2,args=(4,)).start()

部分执行结果:

注:死锁现象只发生在多个进程且存在多个互斥锁的情况在,单个互斥锁的合理使用不会导致死锁现象的发生

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1410238.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

2024.1.22力扣每日一题——最大交换

2024.1.22 题目来源我的题解方法一 暴力法方法一 哈希表贪心方法三 贪心 题目来源 力扣每日一题;题序:670 我的题解 方法一 暴力法 直接暴力对数字中的每两个位置进行交换,然后记录交换后生成数字的最大值 时间复杂度:O( log ⁡…

下拉回显问题案例大全

下拉回显问题案例大全 一、原生js案例 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>下拉框数据回…

13. 高级IO

13. 高级IO 1. 非阻塞 IO1.1 阻塞 IO 与非阻塞 IO 读文件 2. IO 多路复用2.1 何为 IO 多路复用2.2 select()2.3 poll()2.3.1 struct pollfd2.3.2 poll() 返回值2.3.3 示例 3. 异步 IO3.1 O_ASYNC3.2 设置异步 IO 事件的接收进程3.3 示例 4. 优化异步 IO4.1 使用实时信号替换默认…

android:persistent和android:priority的区别,对进程优先级有什么影响?

前言&#xff1a;写的apk因为系统busy给我kill了&#xff0c;(adj 900): kill all background&#xff0c;在AndroidManifest.xml添加android:persistent"true"后&#xff0c;被甲方要求不能这样做&#xff0c;还是得从adj改&#xff0c;把 priority改成1000 android…

ES6.8.6 为索引映射(Mapping)创建自定义分词器,测试分词匹配效果

文章目录 环境创建索引&#xff1a;配置自定义分词器、字段指定分词器自定义分词器参数说明创建索引&#xff1a;custom_analyzer_comment 使用索引中自定义的分词器进行分词分析自定义分词器my_custom_analyzer分词测试&#xff1a;测试中文停用词、英文字母转小写测试敏感词替…

C++1.0

思维导图 提示输入一个字符串&#xff0c;统计该字符中大写&#xff0c;小写字母个数&#xff0c;数字个数&#xff0c;空格个数以及特殊字符个数&#xff0c;要求使用C风格字符串完成 #include <iostream>using namespace std;int main() {cout << "请输入一…

红包六(CTFshow)

jar的逆向&#xff0c;第一次接触 jd逆向工具反编译jar文件 可以直接丢进去看&#xff0c;也可以用jd反汇编工具看 这里提示flag不在这里分析一下这段代码 1. 引入必要的库: java.util.Base64: 用于处理 Base64 编码和解码。java.util.Scanner: 用于从用户输入中读取文本。…

EasyExcel实现下载模板

实体类&#xff1a; package com.aicut.monitor.domain;import com.alibaba.excel.annotation.ExcelIgnore; import com.alibaba.excel.annotation.ExcelIgnoreUnannotated; import com.alibaba.excel.annotation.ExcelProperty; import com.alibaba.excel.annotation.write.s…

浙大AIF发布年度报告:PaaS是金融机构数智化创新“加速器”

“云原生技术体系日趋成熟&#xff0c;云原生分布式PaaS平台对金融业‘用云价值’发挥三个关键作用——提升多云协同效率、降低数字化门槛和新领域试错成本&#xff0c;正成为中国金融机构数字化转型的加速器。”1月25日&#xff0c;浙江大学国际联合商学院院长、金融科技研究院…

2000-2022年中国对225个国家地区进出口数据

2000-2022年中国对225个国家地区进出口数据 1、时间&#xff1a;2000-2022年 2、来源&#xff1a;UN Comtrade联合国贸易数据 3、指标&#xff1a;、年份、年份、报告国家编码、报告国家ISO编码、报告国家、进出口类别编码、进出口类别&#xff08;import进口/export出口&am…

目标检测数据集 - 人脑肿瘤检测数据集下载「包含VOC、COCO、YOLO三种格式」

数据集介绍&#xff1a;人脑肿瘤检测数据集&#xff0c;真实 CT 场景高质量图片数据&#xff0c;涉及人脑 CT 图片数据集丰富&#xff1b;适用实际项目应用&#xff1a;CT 图片场景下人脑肿瘤检测项目&#xff0c;以及作为通用人脑检测数据集场景数据的补充&#xff1b;标注说明…

HTML 炫酷进度条

下面是代码 <!DOCTYPE html> <html><head><meta charset"UTF-8"><title>Light Loader - CodePen</title><style> html, body, div, span, applet, object, iframe, h1, h2, h3, h4, h5, h6, p, blockquote, pre, a, abbr…

docker(第二部分)

来自尚硅谷杨哥 少一点胡思乱想&#xff0c;心中无女人&#xff0c;编码自然神&#xff0c;忘掉心上人&#xff0c;抬手灭红尘。人间清醒&#xff0c;赚钱第一。好好学习&#xff0c;天天向上。听懂六六六。 7.Dokcer容器数据卷 1,&#xff09;坑&#xff1a;容器卷记得加入 …

【并发编程】顺序控制交替输出abc

&#x1f4dd;个人主页&#xff1a;五敷有你 &#x1f525;系列专栏&#xff1a;并发编程⛺️稳重求进&#xff0c;晒太阳 必须先2后1打印 用synchronized package aaa;public class Test2 {static Boolean hasExecutorfalse;public static void main(String[] args) …

TS基础知识点快速回顾(上)

基础介绍 什么是 TypeScript&#xff1f; TypeScript&#xff0c;简称 ts&#xff0c;是微软开发的一种静态的编程语言&#xff0c;它是 JavaScript 的超集。 那么它有什么特别之处呢? js 有的 ts 都有&#xff0c;所有js 代码都可以在 ts 里面运行。ts 支持类型支持&#…

华清远见作业第三十三天——C++(第二天)

思维导图&#xff1a; 题目&#xff1a; 自己封装一个矩形类(Rect)&#xff0c;拥有私有属性:宽度(width)、高度(height)&#xff0c; 定义公有成员函数&#xff1a; 初始化函数&#xff1a;void init(int w, int h) 更改宽度的函数&#xff1a;set_w(int w) 更改高度的函数…

优思学院|如何将AI人工智能融入精益六西格玛?

在当前的制造和服务运营中&#xff0c;许多流程都在一定程度上重复进行&#xff0c;这为实验、学习和持续改进其底层流程提供了机会。直到最近&#xff0c;这些流程的改进大多由人类专家执行。然而&#xff0c;随着包括生成型AI在内的人工智能工具的出现&#xff0c;这一切都在…

阅读go语言工具源码系列之gopacket(谷歌出品)----第一集 DLL的go封装

gopacket项目是google出品的golang第三方库&#xff0c;项目源码地址google/gopacket: Provides packet processing capabilities for Go (github.com) gopacket核心是对经典的抓包工具libpcap(linux平台)和npcap(windows平台)的go封装&#xff0c;提供了更方便的go语言操作接…

JavaScript DOM之Cookie详解

cookie有的地方习惯使用复数形式的cookies&#xff0c;指的是网站为了识别用户的身份或者进行一些必要数据的缓存而使用的技术&#xff0c;它的数据是存在用户的终端上&#xff0c;也就是在浏览器上的。 一、什么是cookie 随着互联网的不断发展各种基于互联网的服务系统逐渐多…

3D点云数据的标定,从搭建环境到点云标定方法及过程,只要有一台Windows笔记本,让你学会点云标定

ptscloudpre: 点云标定准备&#xff1a; 说明&#xff1a; 如下介绍适用windows系统的电脑。apple笔记本同理&#xff0c;但是需要安装MAC版本的anaconda。网址&#xff1a;Free Download | Anaconda可下载对应MAC版本的Anaconda的安装包建议下载2022年或2021年的安装包安装。…