浅谈Python中的全局锁(GIL)问题

浅谈Python中的全局锁(GIL)问题

什么是GIL

GIL 全称 Global Interpreter Lock,意为全局解释器锁。在 Python 中,一次只能执行一个线程,所以为了保证线程安全,引入了 GIL 的概念。GIL 是 Python 解释器中自带的机制,用来保证同一时刻只有一个线程在执行 Python 代码。当一个线程开始执行 Python 代码时,GIL 就会获得并锁住 Python 解释器,保证只有这个线程可以接触到解释器。

GIL的问题

GIL 的存在使得 Python 的多线程执行遭遇瓶颈。虽然 Python 可以使用多进程的方式解决多线程所面临的瓶颈,但多进程的开销相比于多线程会比较大。而且在某些场景下,使用多线程更为方便。

举个例子

例子1

import threading

num = 0

def thread_test():
    global num
    for i in range(1000000):
        num += 1
    print(num)

thread1 = threading.Thread(target=thread_test)
thread2 = threading.Thread(target=thread_test)
thread1.start()
thread2.start()

以上代码会创建两个线程,每个线程都会对 num 进行 1000000 次加 1 操作,并打印最终的计数结果。

但是执行的结果可能出乎想象:

1265211
1394415

实际上,我们想要的结果应该是 2000000。原因就在于 GIL。

在使用多线程执行 Python 代码时,由于 GIL 的存在,每一个时刻只有一个线程在执行 Python 代码。在本例中,两个线程交替获得 GIL,一次只能做一次加法。最终的结果可能超出预期。

例子2

import threading

num = 0

def thread_test():
    global num
    for i in range(1000000):
        num += 1
    print(num)

thread1 = threading.Thread(target=thread_test)
thread2 = threading.Thread(target=thread_test)
thread3 = threading.Thread(target=thread_test)
thread1.start()
thread2.start()
thread3.start()

以上代码会创建三个线程,每个线程都会对 num 进行 1000000 次加 1 操作,并打印最终的计数结果。

执行的结果可能如下:

2324507
2424009
3553736

同样也是超出预期的。

在这个例子中,如果想要结果正确,可以使用 threading.Lock() 对共享变量进行锁定:

import threading

num = 0
lock = threading.Lock()

def thread_test():
    global num
    for i in range(1000000):
        lock.acquire()
        num += 1
        lock.release()
    print(num)

thread1 = threading.Thread(target=thread_test)
thread2 = threading.Thread(target=thread_test)
thread3 = threading.Thread(target=thread_test)
thread1.start()
thread2.start()
thread3.start()

使用 lock 后,每个线程在执行对 num 的加 1 操作前都会尝试获取锁,如果拿到锁后才执行加法操作。这样可以保证在同一时刻只有一个线程对 num 进行操作,最终达到正确的结果。

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:浅谈Python中的全局锁(GIL)问题 - Python技术站

(0)
上一篇 2023年5月19日
下一篇 2023年5月19日

相关文章

  • python 根据正则表达式提取指定的内容实例详解

    以下是详细讲解“Python根据正则表达式提取指定的内容实例详解”的完整攻略,包括正则表达式的基本语法、使用re模块提取指定内容的方法和两个示例说明。 正则表达式的基本语法 正则表达式是一种用于匹配文本的模式。Python中,我们可以使用re模块来处理正则表达式。正则表达式的基本语法如下: 字符:匹指定的字符。 字符集:匹配指定的字符集。 量词:匹配指的数量…

    python 2023年5月14日
    00
  • python爬虫开发之使用python爬虫库requests,urllib与今日头条搜索功能爬取搜索内容实例

    我们来详细讲解”python爬虫开发之使用python爬虫库requests,urllib与今日头条搜索功能爬取搜索内容实例”这个话题,包括以下内容: 什么是Python爬虫? 为什么要使用爬虫库? Requests库和Urllib库的介绍和区别 今日头条搜索功能爬取内容实例 其他爬虫库的简要介绍及应用 1. 什么是Python爬虫? Python爬虫是指使…

    python 2023年5月14日
    00
  • 将不规则的Python多维数组拉平到一维的方法实现

    将不规则的Python多维数组拉平到一维数组是一个常见的问题,下面讲解几种方法来实现它。 方法一:使用itertools.chain()函数 首先,我们可以使用itertools.chain()函数来将多维数组拉平到一维数组。这个函数可以接受多个迭代器作为参数,然后将它们连接起来形成一个单一的迭代器。 示例:将二维数组 [[1, 2, 3], [4, 5, …

    python 2023年6月3日
    00
  • python中openpyxl和xlsxwriter对Excel的操作方法

    下面是详细讲解“python中openpyxl和xlsxwriter对Excel的操作方法”的完整实例教程: 简介 在Python中通过第三方库openpyxl和xlsxwriter可以实现对Excel文件的读写。openpyxl可以读取、修改和写入已有的Excel文件,xlsxwriter是封装了C语言库libxlsxwriter,并提供Python接口,…

    python 2023年5月13日
    00
  • Python打印输出数组中全部元素

    下面是Python打印输出数组中全部元素的完整攻略: 1. 定义数组 在Python中,我们可以使用列表(list)来模拟数组。假设我们要打印输出的数组为[1, 2, 3, 4, 5],可以使用以下代码来定义: arr = [1, 2, 3, 4, 5] # 定义了一个名为arr的列表,其中包含5个整数元素 2. 循环遍历输出数组元素 我们可以使用循环语句来…

    python 2023年6月5日
    00
  • python批量读取txt文件为DataFrame的方法

    下面是“python批量读取txt文件为DataFrame的方法”的完整攻略,包括以下步骤: 步骤一:准备数据 首先需要获得一些示例数据,这些示例数据应该存在于多个.txt文件中。这些文件应该具有相同的格式,可以包含标头和数据,以制表符或其他分隔符分隔。 步骤二:导入必要的库 在使用本方法之前,需要导入pandas库。可以使用以下命令导入pandas: im…

    python 2023年6月2日
    00
  • Python 根据相邻关系还原数组的两种方式(单向构造和双向构造)

    当然,我很乐意为您提供“Python根据相邻关系还原数组的两种方式(单向构造和双向构造)”的完整攻略。以下是详细步骤和示例。 根据相邻关系还原数组的概述 在Python中,有时候我们需要根据相邻关系还原数组。例如,我们有一个长度为n的数组,其中每个元素都是1到n之间的整数,且每个元素都与相邻的元素有关系。现在,我们需要根据这些关系还原原始数组。这个问题可以使…

    python 2023年5月13日
    00
  • python+requests+pytest接口自动化的实现示例

    以下是关于“Python+requests+pytest接口自动化的实现示例”的完整攻略: Python+requests+pytest接口自动化的实现示例 在Python中,我们可以使用requests模块发送HTTP请求,使用pytest框架实现接口自动化测试。以下是Python+requests+pytest接口自动化的实现示例。 安装pytest 在…

    python 2023年5月15日
    00
合作推广
合作推广
分享本页
返回顶部