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

本文主要是介绍QA测试开发工程师面试题满分问答22: (干货)为什么要加锁Lock,举个例子说说,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

加锁原因

下面代码会有什么问题?

import threading
import requests
from queue import Queuedef 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 Queuedef 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+同学入职大厂,【简历优化】、【就业指导】、【模拟/辅导面试】一对一指导

这篇关于QA测试开发工程师面试题满分问答22: (干货)为什么要加锁Lock,举个例子说说的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/977355

相关文章

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

性能测试介绍

性能测试是一种测试方法,旨在评估系统、应用程序或组件在现实场景中的性能表现和可靠性。它通常用于衡量系统在不同负载条件下的响应时间、吞吐量、资源利用率、稳定性和可扩展性等关键指标。 为什么要进行性能测试 通过性能测试,可以确定系统是否能够满足预期的性能要求,找出性能瓶颈和潜在的问题,并进行优化和调整。 发现性能瓶颈:性能测试可以帮助发现系统的性能瓶颈,即系统在高负载或高并发情况下可能出现的问题

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD

字节面试 | 如何测试RocketMQ、RocketMQ?

字节面试:RocketMQ是怎么测试的呢? 答: 首先保证消息的消费正确、设计逆向用例,在验证消息内容为空等情况时的消费正确性; 推送大批量MQ,通过Admin控制台查看MQ消费的情况,是否出现消费假死、TPS是否正常等等问题。(上述都是临场发挥,但是RocketMQ真正的测试点,还真的需要探讨) 01 先了解RocketMQ 作为测试也是要简单了解RocketMQ。简单来说,就是一个分

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

【测试】输入正确用户名和密码,点击登录没有响应的可能性原因

目录 一、前端问题 1. 界面交互问题 2. 输入数据校验问题 二、网络问题 1. 网络连接中断 2. 代理设置问题 三、后端问题 1. 服务器故障 2. 数据库问题 3. 权限问题: 四、其他问题 1. 缓存问题 2. 第三方服务问题 3. 配置问题 一、前端问题 1. 界面交互问题 登录按钮的点击事件未正确绑定,导致点击后无法触发登录操作。 页面可能存在

业务中14个需要进行A/B测试的时刻[信息图]

在本指南中,我们将全面了解有关 A/B测试 的所有内容。 我们将介绍不同类型的A/B测试,如何有效地规划和启动测试,如何评估测试是否成功,您应该关注哪些指标,多年来我们发现的常见错误等等。 什么是A/B测试? A/B测试(有时称为“分割测试”)是一种实验类型,其中您创建两种或多种内容变体——如登录页面、电子邮件或广告——并将它们显示给不同的受众群体,以查看哪一种效果最好。 本质上,A/B测

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta