python并发编程: Python线程安全问题以及解决方案

本文涉及的产品
实时计算 Flink 版,5000CU*H 3个月
检索分析服务 Elasticsearch 版,2核4GB开发者规格 1个月
大数据开发治理平台 DataWorks,不限时长
简介: python并发编程: Python线程安全问题以及解决方案

往期文章:

  1. 并发编程简介
  2. 怎样选择多线程多进程多协程
  3. Python速度慢的罪魁祸首,全局解释器锁GIL
  4. 使用多线程,Python爬虫被加速10倍
  5. Python实现生产者消费者爬虫

线程安全概念介绍

线程安全指某个函数、函数库在多线程环境中被调用时,能够正确地处理多个线程之间的共享变量,使程序功能正确完成。由于线程的执行随时会发生切换,就造成了不可预料的结果,出现线程不安全。

上图展示的是一个取钱的过程,每次取钱,先进行if判断,然后再减去金额。线程1执行到if判断完,就被切换到线程2了。 此时,线程2也进入到了if中又被切换到线程1,线程1继续执行下去,减去金额,取到了钱。切换到线程2,也减去金额,取到了钱,显然就有问题了。 银行亏了600块。

Lock用于解决线程安全问题

try-finally模式

import threading

lock = threading.Lock()
lock.acquire()
try:
    #do something
finally:
    lock.release()

with模式(更常用)

import threading

lock = threading.Lock()
whit lock:
  #do something

线程锁使用实例

import threading
import time
from loguru import logger

lock = threading.Lock()

class Account:

    def __init__(self,balance) -> None:
        self.balance = balance

def draw(account:Account,amount):
    with lock:
        if account.balance >= amount:
            time.sleep(1)
            logger.info("{}取钱成功".format(threading.current_thread().name))
            account.balance -= amount
            logger.info("线程{},{}余额".format(threading.current_thread().name,account.balance))
        else:
            logger.error("{}取钱失败,余额不足!".format(threading.current_thread().name))        

if __name__ == "__main__":
    account = Account(1000)
    task1 = threading.Thread(target=draw,args=(account,800),name="task1")
    task2 = threading.Thread(target=draw,args=(account,800),name="taks2")

    task1.start()
    task2.start()

执行结果如下:

目录
相关文章
聊聊python多线程与多进程
为什么要使用多进程与多线程呢? 因为我们如果按照流程一步步执行任务实在是太慢了,假如一个任务就是10秒,两个任务就是20秒,那100个任务呢?况且cpu这么贵,时间长了就是浪费生命啊!一个任务比喻成一个人,别个做高铁,你做绿皮火车,可想而知!接下来我们先看个例子:
|
4天前
|
安全 算法 Java
Java中的并发编程技术:解锁高效多线程应用的秘密
Java作为一种广泛应用的编程语言,其并发编程技术一直备受关注。本文将深入探讨Java中的并发编程,从基本概念到高级技巧,帮助读者更好地理解并发编程的本质,并学会如何在多线程环境中构建高效可靠的应用程序。
|
5天前
|
分布式计算 安全 Java
Python 多线程
Python 多线程
14 0
|
6天前
|
数据挖掘 调度 Python
线程:并发编程基石
线程:并发编程基石
|
6天前
|
数据采集 存储 C++
单线程 vs 多进程:Python网络爬虫效率对比
本文探讨了Python网络爬虫中的单线程与多进程应用。单线程爬虫实现简单,但处理速度慢,无法充分利用多核CPU。而多进程爬虫通过并行处理提高效率,更适合现代多核架构。代码示例展示了如何使用代理IP实现单线程和多进程爬虫,显示了多进程在效率上的优势。实际使用时还需考虑代理稳定性和反爬策略。
单线程 vs 多进程:Python网络爬虫效率对比
|
7天前
|
安全 Java 程序员
Java 多线程系列Ⅵ(并发编程的六大组件)
Java 多线程系列Ⅵ(并发编程的六大组件)
|
7天前
|
Python
深度解析Python中的多线程编程
深度解析Python中的多线程编程
30 1
|
8天前
|
安全 Java 调度
Java多线程基础-14:并发编程中常见的锁策略(二)
这段内容介绍了互斥锁和读写锁的概念以及它们在多线程环境中的应用。互斥锁仅允许进入和退出代码块时加锁和解锁,而读写锁则区分读和写操作,允许多个线程同时读但写时互斥。
22 0
|
8天前
|
算法 安全 Java
Java多线程基础-14:并发编程中常见的锁策略(一)
乐观锁和悲观锁是并发控制的两种策略。悲观锁假设数据容易产生冲突,因此在读取时即加锁,防止其他线程修改,可能导致效率较低。
21 0
|
8天前
|
算法 Java
Java多线程基础-13:一文阐明死锁的成因及解决方案
死锁是指多个线程相互等待对方释放资源而造成的一种僵局,导致程序无法正常结束。发生死锁需满足四个条件:互斥、请求与保持、不可抢占和循环等待。避免死锁的方法包括设定加锁顺序、使用银行家算法、设置超时机制、检测与恢复死锁以及减少共享资源。面试中可能会问及死锁的概念、避免策略以及实际经验。
17 1