threading.local()使用与原理剖析

前言

  还是第一次摘出某个方法来专门写一篇随笔,哈哈哈。

  为什么要写这个方法呢?因为它确实太重要了,包括后期的Flask框架源码中都有它的影子。

  那么我们就来瞄一眼这个东西是啥吧。

作用

  在Python官方中文文档中(Python3.8.4),对它的介绍其实并不是很详细

  其实他的功能非常简单,如下:

  在一个全局的容器中可以存放一些线程独有的数据,这些数据应是某一线程私有的,是除了本线程外的其他线程访问不到的。

  举个例子,例如你用迅雷下载的时候每条线程的下载进度不一样,我们需要将下载进度这个数据存储起来,怎么存储?自己创建一个具有线程安全性的列表或字典?那太麻烦了,请记住一点,数据怎么存不重要,关键是要方便取,取的快,取的准才是王道,所以直接用thrading.local()方法即可。

  我来画一张吧,灵魂画师上线,能用图描述绝对不打字。

基本使用

  threading.local():可以实例化出一个寄存柜,这个寄存柜是全局化的

  寄存柜对象.你想放的东西名字 = 东西:你可以在线程中这样放入一个独有的物件

  寄存柜对象.你放过的东西的名字:这样,你就可以将你存放进的东西拿出来。注意,只有你才能拿出来,其他人拿不了。

import threading
import time def operate(name):
"""操作"""
print("三个月后,{0}忽然想起了自己有件东西放在柜子里,决定取出来".format(name))
print("最后他从自己的柜子里取出来了:",Locker.article) def people(article):
name = threading.current_thread().getName() # 获取线程名
Locker.article = article # 这就表示将私有的的一件物品放在了寄存柜里
print("{0}将{1}放在了寄存柜里...".format(name,article))
time.sleep(3) # 过了三个月,这期间发生了很多事
operate(name) if __name__ == '__main__':
Locker = threading.local() # 好了寄存柜放在全局了
t1 = threading.Thread(target=people,args=("一封情书",),name="小王")
t2 = threading.Thread(target=people,args=("一双臭袜子",),name="小李")
t1.start()
t2.start()
t1.join()
t2.join() # ==== 执行结果 ==== """
小王将一封情书放在了寄存柜里...
小李将一双臭袜子放在了寄存柜里...
三个月后,小王忽然想起了自己有件东西放在柜子里,决定取出来
最后他从自己的柜子里取出来了: 一封情书
三个月后,小李忽然想起了自己有件东西放在柜子里,决定取出来
最后他从自己的柜子里取出来了: 一双臭袜子
"""

原理分析

  我们可以自己做一个全局字典,来实现与这个类似的功能,但是使用起来肯定不太方便(它实际上本质就是字典嵌套进行存储的):

import threading
import time Locker = {} # 好了寄存柜放在全局了
"""
{
线程id:{"article":"存放的具体物品"},
线程id:{"article":"存放的具体物品"},
线程id:{"article":"存放的具体物品"},
}
""" def operate(name):
"""操作"""
print("三个月后,{0}忽然想起了自己有件东西放在柜子里,决定取出来".format(name))
ident = threading.get_ident() # 获取线程ID
print("最后他从自己的柜子里取出来了:",Locker[ident]["article"]) def people(article):
name = threading.current_thread().getName() # 获取线程名
ident = threading.get_ident() # 获取线程ID
Locker[ident] = {} # 创建了一个寄存柜的小格子
Locker[ident]["article"] = article # 这就表示将私有的的一件物品放在了寄存柜里
print("{0}将{1}放在了寄存柜里...".format(name,article))
time.sleep(3) # 过了三个月,这期间发生了很多事
operate(name) if __name__ == '__main__': t1 = threading.Thread(target=people,args=("一封情书",),name="小王")
t2 = threading.Thread(target=people,args=("一双臭袜子",),name="小李")
t1.start()
t2.start()
t1.join()
t2.join() # ==== 执行结果 ==== """
小王将一封情书放在了寄存柜里...
小李将一双臭袜子放在了寄存柜里...
三个月后,小李忽然想起了自己有件东西放在柜子里,决定取出来
最后他从自己的柜子里取出来了: 一双臭袜子
三个月后,小王忽然想起了自己有件东西放在柜子里,决定取出来
最后他从自己的柜子里取出来了: 一封情书
"""

尝试自己做出一个寄存柜

  这样做是不是太麻烦了?我们可以自定义一个类,让它变得更加简单,如同原本的使用方法一样。其实下面代码中采取的方法也是threading.local()所采取的方法。

import threading
import time class MyLocker(object):
cabinet = {} # 柜子
"""
{
线程id:{"article":"存放的具体物品"},
线程id:{"article":"存放的具体物品"},
线程id:{"article":"存放的具体物品"},
}
""" def __getattr__(self, item):
"""当访问属性不存在时触发"""
ident = threading.get_ident()
return MyLocker.cabinet[ident][item] def __setattr__(self, key, value):
"""试图用 . 去设置属性时触发"""
ident = threading.get_ident()
if ident in MyLocker.cabinet: # 如果在柜子里这重新赋值
MyLocker.cabinet[ident][key] = value
else:
MyLocker.cabinet[ident] = {key: value} # 如果不在则做一个小格子,并且把物件存放进来 def operate(name):
"""操作"""
print("三个月后,{0}忽然想起了自己有件东西放在柜子里,决定取出来".format(name))
print("最后他从自己的柜子里取出来了:", Locker.article) def people(article):
name = threading.current_thread().getName() # 获取线程名
Locker.article = article # 这就表示将私有的的一件物品放在了寄存柜里
print("{0}将{1}放在了寄存柜里...".format(name, article))
time.sleep(3) # 过了三个月,这期间发生了很多事
operate(name) if __name__ == '__main__':
Locker = MyLocker() # 好了寄存柜放在全局了
t1 = threading.Thread(target=people, args=("一封情书",), name="小王")
t2 = threading.Thread(target=people, args=("一双臭袜子",), name="小李")
t1.start()
t2.start()
t1.join()
t2.join() # ==== 执行结果 ==== """
小王将一封情书放在了寄存柜里...
小李将一双臭袜子放在了寄存柜里...
三个月后,小王忽然想起了自己有件东西放在柜子里,决定取出来
最后他从自己的柜子里取出来了: 一封情书
三个月后,小李忽然想起了自己有件东西放在柜子里,决定取出来
最后他从自己的柜子里取出来了: 一双臭袜子
"""

threading.local()使用与原理剖析的更多相关文章

  1. threading.local作用及原理

    先看下应用: import threading from threading import local import time obj = local() def task(i): obj.xxxxx ...

  2. flask上下文管理相关 - threading.local 以及原理剖析

    threading.local 面向对象相关: setattr/getattr class Foo(object): pass obj = Foo() obj.x1 = 123 # object.__ ...

  3. 多线程threading.local的作用及原理?

    1.示例代码 import time import threading v = threading.local() def func(arg): # 内部会为当前线程创建一个空间用于存储:phone= ...

  4. 03 flask源码剖析之threading.local和高级

    03 threading.local和高级 目录 03 threading.local和高级 1.python之threading.local 2. 线程唯一标识 3. 自定义threading.lo ...

  5. ASP.NET Core 运行原理剖析2:Startup 和 Middleware(中间件)

    ASP.NET Core 运行原理剖析2:Startup 和 Middleware(中间件) Startup Class 1.Startup Constructor(构造函数) 2.Configure ...

  6. 【Xamarin挖墙脚系列:Xamarin.IOS机制原理剖析】

    原文:[Xamarin挖墙脚系列:Xamarin.IOS机制原理剖析] [注意:]团队里总是有人反映卸载Xamarin,清理不完全.之前写过如何完全卸载清理剩余的文件.今天写了Windows下的批命令 ...

  7. NameNode和SecondaryNameNode工作原理剖析

    NameNode和SecondaryNameNode工作原理剖析 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.NameNode中的元数据是存储在那里的? 1>.首先,我 ...

  8. 开源 serverless 产品原理剖析 - Kubeless

    背景 Serverless 架构的出现让开发者不用过多地考虑传统的服务器采购.硬件运维.网络拓扑.资源扩容等问题,可以将更多的精力放在业务的拓展和创新上. 随着 serverless 概念的深入人心, ...

  9. 自定义threading.local

    1.threading相关. # Author:Jesi # Time : 2018/12/28 14:21 import threading import time from threading i ...

随机推荐

  1. Java实现 LeetCode 516 最长回文子序列

    516. 最长回文子序列 给定一个字符串s,找到其中最长的回文子序列.可以假设s的最大长度为1000. 示例 1: 输入: "bbbab" 输出: 4 一个可能的最长回文子序列为 ...

  2. Java实现网格中移动字母

    2x3=6个方格中放入ABCDE五个字母,右下角的那个格空着.如图[1.jpg]所示. 和空格子相邻的格子中的字母可以移动到空格中,比如,图中的C和E就可以移动,移动后的局面分别是: A B D E ...

  3. Windows10开启Ubuntu子系统并搭建Docker环境

    前言 很早就听说微软有个基于Ubuntu的子系统,一直也没机会尝试一下,之前也只是用VMware安装,但是还要单独安装软件,安装镜像,一点都不fit,所以就瞎折腾下(也是因为最近有空). 搭建Ubun ...

  4. postman接口超时设置,用于debug等断点调试

    Settings->General->Request Timeout in ms(0 for infinity):设置请求超时的时间,默认为0

  5. iOS -实现UIView圆角显示的方法

    添加一个UIView时,默认直角显示有时对于有强迫症的患者还真有点不舒服! eg: 怎么实现UIview的圆角显示呢? 首先包含一个头文件: #import <QuartzCore/Quartz ...

  6. 如何解决ubuntu 12.04重启后出现waiting for network configuration和网络标志消失问题

    如何解决ubuntu 12.04重启后出现waiting for network configuration和网络标志消失问题 作为菜鸟的我在学着设置网络后,重启电脑后显示 waiting forne ...

  7. InnoDB存储引擎的事务

    事务的任务是保证一系列更新语句的原子性,锁的任务是解决并发访问可能导致的数据不一致问题.如果事务与事务之间存在并发操作,此时可以通过隔离级别实现事务的隔离性,从而实现数据的并发访问. 1 原子性(At ...

  8. C#winform单线程事例与多线程事例

    通过例子编写,用winform编写的,讲解单线程与多线程使用,用于异步加载数据,界面不会卡死,数据在后台默认加载,给用户更好的体验.稍后会附加完整代码. 1.先不用线程,显示一个求和,计算过程中要停留 ...

  9. 浏览器端如何使用VConsole.js调试代码?

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  10. python2.7 函数的参数学习

    1.默认参数 默认参数可以简化函数的调用. 设置默认参数时,有几点要注意: 一.必选参数在前,默认参数在后,否则Python的解释器会报错. 二.当函数有多个参数时,把变化大的参数放前面,变化小的参数 ...