QA测试开发工程师面试题满分问答22: (干货)为什么要加锁Lock,举个例子说说

加锁原因

下面代码会有什么问题?

import threading
import requests
from queue import Queue

def make_request(url, params, results_queue):
    response = requests.get(url, params=params)
    result = {
        'url': url,
        'params': params,
        'response': response.text
    }
    results_queue.put(result)

# 定义接口 URL
url = "https://api.example.com/endpoint"

# 定义要传递给接口的不同参数列表
params_list = [
    {'param1': 'value1'},
    {'param2': 'value2'},
    {'param3': 'value3'}
]

# 创建线程列表和结果队列
threads = []
results_queue = Queue()

# 遍历参数列表,为每个参数创建一个线程并启动
for params in params_list:
    thread = threading.Thread(target=make_request, args=(url, params, results_queue))
    threads.append(thread)
    thread.start()

# 等待所有线程执行完毕
for thread in threads:
    thread.join()

# 从结果队列中获取结果
results = []
while not results_queue.empty():
    result = results_queue.get()
    results.append(result)

# 打印结果
for result in results:
    print(f"URL: {result['url']}, Params: {result['params']}, Response: {result['response']}")

results_queue.put(result),这里会有子线程并发写进同一个共享变量中,可能会数据不一致

加锁Lock

加个lock避免共享变量读写冲突,demo

import threading
import requests
from queue import Queue

def get_GTP4_result(url, results_dict):
    # 省略部分代码

    results_dict[url] = response

# 定义要传递给接口的不同参数列表
url_list = [
    "http://1.jpg",
    "https://2.jpg"
]

# 创建线程列表和结果字典
threads, results_dict = [], {}

# 创建锁对象
lock = threading.Lock()

# 定义线程函数
def thread_function(url):
    response = get_GTP4_result(url)
    with lock:
        results_dict[url] = response

# 遍历参数列表,为每个参数创建一个线程并启动
for url in url_list:
    thread = threading.Thread(target=thread_function, args=(url,))
    threads.append(thread)
    thread.start()

# 等待所有线程执行完毕
for thread in threads:
    thread.join()

# 按照URL的顺序获取结果
results = [results_dict[url] for url in url_list]

# 打印结果
for result in results:
    print(f"Response: {result}")

加lock,具体有哪些好处

加锁(Lock)用于保护共享资源的访问,确保线程安全性。在多线程环境中,多个线程同时访问和修改共享变量可能导致数据竞争和不一致的结果。加锁可以解决这个问题,并提供以下好处:

  1. 数据一致性:加锁可以防止多个线程同时对共享变量进行读写操作,从而避免数据的不一致性。当一个线程获得锁后,其他线程需要等待锁的释放,确保每次只有一个线程能够修改共享变量。

  2. 线程安全性:通过加锁,可以保证对共享变量的操作是线程安全的。线程安全意味着多个线程并发地访问共享资源时,不会出现竞争条件、数据损坏或不一致的情况。

  3. 避免竞争条件:通过使用锁,可以防止多个线程同时执行对共享变量的修改操作,从而避免竞争条件的发生。竞争条件是指多个线程在执行过程中产生的不确定性结果,可能导致程序的错误行为。

  4. 保护临界区:加锁可以将一段代码标记为临界区,只允许一个线程进入执行。这样可以确保在该临界区内对共享变量的操作是原子的,不会受到其他线程的干扰。

  5. 同步线程执行顺序:使用锁可以控制多个线程的执行顺序,确保它们按照预期的顺序执行。通过在合适的地方加锁和释放锁,可以实现线程之间的同步和协调。

python的多线程如何加锁和释放锁,有几种方式

在 Python 中,可以使用 threading 模块提供的锁对象 Lock 来实现线程的加锁和释放锁。常用的加锁和释放锁的方法有以下几种:

  1. 使用 Lock 对象的 acquire() 和 release() 方法:

    • acquire() 方法用于获取锁,如果锁已经被其他线程获取,则当前线程会被阻塞,直到锁被释放。
    • release() 方法用于释放锁,将锁的状态设置为可获取,允许其他线程获取该锁。
    import threading
    
    # 创建锁对象
    lock = threading.Lock()
    
    # 在需要加锁的代码块中使用锁
    lock.acquire()
    try:
        # 执行需要加锁的操作
        # ...
    finally:
        # 释放锁
        lock.release()
    
  2. 使用 with 语句简化锁的使用:
    Lock 对象也支持上下文管理器协议,可以使用 with 语句来自动获取和释放锁。

    import threading
    
    # 创建锁对象
    lock = threading.Lock()
    
    # 在需要加锁的代码块中使用锁
    with lock:
        # 执行需要加锁的操作
        # ...
    
  3. 使用 RLock 对象实现可重入锁:
    RLock 是 Lock 的子类,支持同一个线程多次获取锁而不会造成死锁。如果在一个线程中多次获取 Lock 对象,该线程会一直处于阻塞状态,因为它已经持有了该锁。而 RLock 对象允许同一个线程多次获取锁,并通过相同次数的释放操作来释放锁。

    import threading
    
    # 创建可重入锁对象
    rlock = threading.RLock()
    
    # 在需要加锁的代码块中使用可重入锁
    rlock.acquire()
    try:
        # 执行需要加锁的操作
        # ...
    finally:
        # 释放锁
        rlock.release()
    

这些方法都可以保证线程安全性,并避免多个线程同时访问共享资源。选择使用哪种方法取决于具体的需求和场景。通常情况下,推荐使用 with 语句来简化锁的使用,因为它可以自动处理锁的获取和释放,避免忘记释放锁的情况发生。如果需要支持同一个线程多次获取锁,可以使用 RLock 对象。

python多线程还有没有其他锁呢

除了常用的 Lock 和 RLock,Python 的 threading 模块还提供了其他类型的锁,用于不同的场景和需求。以下是一些常见的锁类型:

  1. Semaphore(信号量):信号量是一种允许多个线程同时访问临界区的锁机制。它维护一个内部计数器,线程在进入临界区之前必须先获得信号量的许可,每次许可获取后计数器减一。当计数器为零时,其他线程需要等待。可以使用 Semaphore 对象的 acquire() 和 release() 方法来控制许可的获取和释放。

    import threading
    
    # 创建信号量对象,指定许可的数量
    semaphore = threading.Semaphore(2)  # 允许两个线程同时访问临界区
    
    # 在需要访问临界区的代码块中使用信号量
    semaphore.acquire()
    try:
        # 执行需要访问的临界区操作
        # ...
    finally:
        # 释放信号量
        semaphore.release()
    
  2. Event(事件):事件是一种线程间的通信机制,用于一个或多个线程等待某个事件的发生。它有两种状态:已设置和未设置。线程可以等待事件的设置,也可以通过设置事件来通知其他线程。Event 对象的 wait() 方法用于等待事件的设置,set() 方法用于设置事件,clear() 方法用于清除事件。

    import threading
    
    # 创建事件对象
    event = threading.Event()
    
    # 在等待事件发生的线程中使用事件
    event.wait()
    
    # 在设置事件的线程中使用事件
    event.set()
    
    # 在清除事件的线程中使用事件
    event.clear()
    
  3. Condition(条件):条件是一种复杂的锁机制,用于线程间的协调和通信。它提供了一个线程可以等待的条件,并且可以通过 wait()notify() 和 notifyAll() 方法来实现线程间的协调。Condition 对象基于底层的 Lock 对象来提供线程安全性。

    import threading
    
    # 创建条件对象
    condition = threading.Condition()
    
    # 在等待条件满足的线程中使用条件
    condition.wait()
    
    # 在满足条件的线程中使用条件
    condition.notify()
    
    # 在满足条件的线程中使用条件,通知所有等待的线程
    condition.notifyAll()
    

这些锁类型提供了更多的灵活性和功能,可以根据具体的需求选择适合的锁机制。在多线程编程中,根据线程间的协作和通信需求来选择合适的锁类型非常重要,以确保线程安全和正确的并发行为。

   三段头部互联网大厂测开经历,辅导过25+同学入职大厂,【简历优化】、【就业指导】、【模拟/辅导面试】一对一指导

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

新型AI Stable Artisan横空出世?

StabilityAI宣布推出Stable Artisan 前言 就在今天,Stability AI宣布推出 Stable Artisan,让更广泛的受众能够使用 Stability AI 的 Developer Platform API 功能。Stable Artisan 具有他们的高级型号,例如 Stable Diffusion 3、Stable Video…

宇宙数字扩展全球业务,设立欧洲和亚洲分支机构

2024年4月18日 宇宙数字蚂蚁矿机今日宣布在欧洲和亚洲设立新的分支机构,这一举措旨在进一步强化公司的全球服务网络,提供更地道的客户支持和更快的物流服务,以提升用户满意度。 新的分支机构将位于欧洲和亚洲的战略性城市,为当地客户提供更快速和更便捷的服务。通过本地化的客…

记一次springboot jpa更新复杂几何类型报错Only simple geometries should be used

问题: 更新数据时, 几何字段MultiPolygon类型时报错; java.lang.IllegalStateException: Only simple geometries should be used 几何字段Point类型时不报错; 新增时字段存在MultiPolygon不报错。 查看日志可知,…

重磅!一款实景三维建模软件官宣免费开放

近日,RealityCapture推出了1.4版本。新版本除了常规功能的新增和修复外,更为重磅的是推出了新的定价模式!RealityCapture1.4版本官宣。将对学生、教育工作者、业余爱好者和年收入低于100万美元的公司免费提供,而且还是所有功能&am…

Java Swing游戏开发学习27

内容来自RyiSnow视频讲解 这一节讲的是Equip & Use Items装备与使用物品。 前言 实现捡起物品、切换武器装备、使用物品。 修复问题 当光标在物品栏(背包)中移动到没有物品的格子中的时候,使装备介绍子窗口不可见,反之可见…

CSS-浮动

float (浮动) 作用:盒子的顶点是一样的,具备行内块的特征,能设置宽高 属性:float 属性值:left 浮动在网页左边 right 浮动在网页右边 .a{width: 100px;height: 100px;float:left;background-color: red;}.b…

云手机:海外舆情监控的新工具

在数字化时代,海外舆情监控对于企业、品牌和政府机构来说,已经变得至关重要。传统的舆情监控方法往往受限于地域、设备和技术,而云手机的出现,为海外舆情监控带来了全新的解决方案。 一、云手机与海外舆情监控的完美结合 云手机作…

一个开源即时通讯源码

一个开源即时通讯源码 目前已经含服务端、PC、移动端即时通讯解决方案,主要包含以下内容。 服务端简介 不要被客户端迷惑了,真正值钱的是服务端, 服务是采用Java语言开发,基于spring cloud微服务体系开发的一套即时通讯服务端。…

华为eNSP学习—IP编址

IP编址 IP编址子网划分例题展示第一步:机房1的子网划分第二步:机房2的子网划分第三步:机房3的子网划分IP编址 明确:IPv4地址长度32bit,点分十进制的形式 ip地址构成=网络位+主机位 子网掩码区分网络位和主机位 学此篇基础: ①学会十进制与二进制转换 ②学会区分网络位和…

Python批量备份华为设备配置到FTP服务器

Excel表格存放交换机信息: 备份文件夹效果图: Windows系统配置计划任务定时执行python脚本: Program/script:C:\Python\python.exe Add arguments (optional): D:\Python_PycharmProjects\JunLan_pythonProje…

能源系统升级BACnet IP分布式I/O边缘模块深度整合

能源管理系统(EMS)的高效运行成为了实现绿色建筑、节能减排的关键。而BACnet IP分布式远程I/O模块作为这一系统中的重要组件,正发挥着不可小觑的作用。本文将以某大型商业综合体为例,探讨BACnet IP I/O模块如何在能源管理中大显身手。 商业综合体涵盖办公…

揭秘APP广告变现:轻松赚取额外收入!

在移动应用(APP)的世界中,变现能力是衡量一个应用成功与否的关键指标之一。无论是个人开发者还是企业团队,如何通过应用创造收入,始终是一个备受关注的话题。今天,我们将深入探讨APP广告变现的路径&#xf…

【ElementUI -- 优化小技巧系列】 -- el-tree 节点内容过长优化 以及选中默认节点

在使用elementui过程中经常碰到关于样式的问题,我曾经很喜欢通过类名修改css样式来做,其实原生封装的elementui库的样式对于普通开发来说已经足够了,通过类名修改css只会让组件臃肿难以维护,现在真的越来越怕写css,经常…

镜舟科技亮相2024中国移动算力网络大会、Qcon、DTC等多项活动

在刚刚过去的 4 月份,镜舟科技受邀参与一系列技术交流活动,与移动云、金科创新社、infoQ、墨天轮、开科唯识等媒体及合作伙伴展开积极交流,并分享其在数据技术、金融等垂直行业领域的创新实践,从产业侧、业务侧、技术侧洞察需求、…

基于springboot+mybatis+vue的项目实战之页面参数传递

如图所示&#xff0c;删除操作可以用按钮实现&#xff0c;也可以用超链接来实现。 1、第一种情况&#xff0c;用按钮实现。 html页面相关&#xff1a; <button type"button" click"deleteId(peot.id)">删除</button> <script>new Vue(…

Redis 支持的 Java 客户端都有哪些?

Redis 是一种高性能的键值存储系统&#xff0c;它以其快速、灵活和可扩展的特性而闻名。在 Java 开发中&#xff0c;与 Redis 交互的方式通常是通过使用 Redis 的 Java 客户端。 这些客户端提供了访问 Redis 数据库的接口&#xff0c;使开发人员能够在 Java 应用程序中轻松地使…

活动报名 | 某头部股份制银行,构建实时指标平台的最佳实践

&#x1f449;欢迎到镜舟科技公众号报名了解研讨会 数字化转型不仅是一场技术革命&#xff0c;更是企业决策模式的革新。在这一过程中&#xff0c;数据成为企业最宝贵的资产&#xff0c;实时数据分析对企业决策至关重要。 随着业务复杂性增加&#xff0c;各业务部门数据指标越…

【概率论基础】 一篇文章缕清概率论常见概念关系

碎碎念&#xff1a;再写CSDN之前有一小段时间写数模公众号的经历&#xff0c;但是公众号看的人实在太少了&#xff0c;而且排版和公式、代码编辑都没有CSDN这么方便&#xff0c;所以坚持一算时间就没有更新了。公众号大多写的是概念性的基础&#xff0c;稍加修改搬到咱们的主战…

人人都是开发者?Baidu Comate智能代码助手改变你传统的编程之路

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 文章目录 引入一、人人都是开发者二、Baidu Comate 智能编码助手2.1 Baidu Comate 是什么&#xff1f;2.2 Baidu Comate 支持那…

5到15秒片头音乐200款,30秒片头音效音乐大全

一、素材描述 本套音乐音效素材&#xff0c;大小2.88G&#xff0c;13个压缩文件。 二、素材目录 200个5到15秒的片头音乐.zip 30秒片头-1.zip 30秒片头-2.zip 30秒片头-3.zip 30秒片头-4.zip 30秒片头-5.zip 30秒片头-6.zip 30秒片头-7.zip 30秒片头-8.zip 30秒片头…
最新文章