zoukankan      html  css  js  c++  java
  • Flask快速入门(12) — local与偏函数

    多个线程修改同一个数据,复制多份变量给每个线程用,比如我们用的request,怎样让不同的请求有自己的request。就是为每个线程开辟一块空间进行数据存储

    不用threading.local

    from threading import Thread
    import time
    ctx = -1
    def task(arg):
        global ctx
        ctx = arg
        # time.sleep(2)
        print(ctx)
    
    for i in range(10):
        t = Thread(target=task,args=(i,))
        t.start()
    

    使用threading.local

    from threading import Thread,local
    # 特殊的对象
    ctx = local()
    def task(arg):
        ctx.value = arg
        # time.sleep(2)
        print(ctx.value)
    for i in range(10):
        t = Thread(target=task,args=(i,))
        t.start()
    

    通过字典自定义threading.local(函数)

    from threading import Thread, get_ident
    storage = {}
    def set(k, v):
        ident = get_ident()  # 获取线程id
        if ident in storage:
            storage[ident][k] = v
        else:
            storage[ident] = {k: v}
    def get(k):
        ident = get_ident()
        return storage[ident][k]
    def task(arg):
        set('val', arg)
        v = get('val')
        print(v)
    for i in range(10):
        t = Thread(target=task, args=(i,))
        t.start()
    

    面向对象版

    from threading import Thread,get_ident
    class Local(object):
        storage = {}
        def get(self,k):
            ident = get_ident()
            return Local.storage[ident][k]
    
        def set(self,k,v):
            ident = get_ident()
            if ident in Local.storage:
                Local.storage[ident][k] = v
            else:
                Local.storage[ident] = {k:v}
    obj = Local()
    def task(arg):
        obj.set('val',arg)
        v = obj.get('val')
        print(v)
    
    for i in range(10):
        t = Thread(target=task,args=(i,))
        t.start()
    

    通过setattr和getattr实现

    from threading import Thread,get_ident
    
    class Local(object):
        storage = {}
        def __setattr__(self, key, value):
            ident = get_ident()
            if ident in self.storage:
                self.storage[ident][key] = value
            else:
                self.storage[ident] = {key:value}
    
        def __getattr__(self, key):
            ident = get_ident()
            return self.storage[ident][key]
    obj = Local()
    def task(arg):
        obj.value = arg
        print(obj.value)
    
    for i in range(10):
        t = Thread(target=task,args=(i,))
        t.start()
    

    每个对象有自己的存储空间(字典)

    from threading import Thread,get_ident
    
    class Local(object):
        def __init__(self):
            object.__setattr__(self,'storage',{})
            # self.storage = {}
        def __setattr__(self, key, value):
            ident = get_ident()
            if ident in self.storage:
                self.storage[ident][key] = value
            else:
                self.storage[ident] = {key:value}
        def __getattr__(self, key):
            ident = get_ident()
            return self.storage[ident][key]
    obj = Local()
    def task(arg):
        obj.value = arg
        print(obj.value)
    for i in range(10):
        t = Thread(target=task,args=(i,))
        t.start()
    

    兼容线程和协程(源码到request中去看,看local的__getattr__,setattr)

    # 如果是协程get_ident获取的是协程id
    try:
        from greenlet import getcurrent as get_ident  
    except Exception as e:
        from threading import get_ident
    from threading import Thread
    import time
    class Local(object):
        def __init__(self):
            object.__setattr__(self,'storage',{})
        def __setattr__(self, k, v):
            ident = get_ident()
            if ident in self.storage:
                self.storage[ident][k] = v
            else:
                self.storage[ident] = {k: v}
        def __getattr__(self, k):
            ident = get_ident()
            return self.storage[ident][k]
    obj = Local()
    def task(arg):
        obj.val = arg
        obj.xxx = arg
        print(obj.val)
    for i in range(10):
        t = Thread(target=task,args=(i,))
        t.start()
       
    

    partial偏函数

    # 偏函数的第二个部分(可变参数),按原有函数的参数顺序进行补充,参数将作用在原函数上,最后偏函数返回一个新函数
    from functools import partial
    def test(a,b,c,d):
        return a+b+c+d
    
    tes=partial(test,1,2)
    print(tes(3,4))
    
  • 相关阅读:
    “xmlns:d="http://schemas.microsoft.com/expression/blend/2008"” 命名空间不能被解析
    mysql 连接命令 表管理 ,克隆表,临时表,字符串属性,设定语句间的分隔符
    视图
    NULL值比较,两个列的合并,列值按条件替换。
    词性和 句子成份
    设置用户帐号只能一个地方登录
    数字的格式化,异常处理
    接口,构造器与垃圾回收机制,对像的生命周期。非静态方法与静态方法。
    继承,多态..
    JavaScript —— 常用数据类型隐式转换
  • 原文地址:https://www.cnblogs.com/863652104kai/p/11688602.html
Copyright © 2011-2022 走看看