flask 3 上下文管理、threading.local相关、基础知识

Flask day3 上下文管理、threading.local相关、基础知识

内容回顾

1.Linux命令

cd
vim
mkdir
ls
touch
cat
sed

2.面向对象:特殊方法

obj['x']=123  # 会执行__setitem__方法
obj.x=123  # 会执行setattr方法
obj + 123

3.functools

# 自动传参
from functools import partial

def func(a, b, c):
    return a + b + c


v1 = func(10, 20, 3)
print(v1)

new_func = partial(func, 10, 20)

v2 = new_func(3)
print(v2)

4.认识的装饰器?应用场景

应用:
	-flask:路由、brfore_request
	-django:csrf,缓存,用户登录

5.Flask

  1. 蓝图
  2. session原理

今日内容

1.threading.local

作用:为每个线程开辟一块空间进行数据存储

from threading import local
from threading import Thread
from threading import get_ident
import time

# 特殊的对象,让对象之间相互独立
obj = local()
# obj = -1

def task(arg):
    # 对象.value = 1/2/3/4/5
    print(get_ident())
    obj.value = arg
    time.sleep(2)
    print(obj.value)


for i in range(10):
    t = Thread(target=task, args=(i,))
    t.start()

>>>>>>>>
3920
11004
740
9880
7396
13304
10220
10492
7928
6904


自己通过字典创建一个类似于threading.local的东西

1.函数方式

{
    3920:{val1:0}
    11004:{val1:1}
    740:{val1:3}
    9880:{}
    7396:{}
}



storage = {}


def set(k, v):
    ident = get_ident()
    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)
    print(storage)
    time.sleep(2)
    v = get('val')
    print(v)


for i in range(10):
    t = Thread(target=task, args=(i,))
    t.start()
    

2.面向对象方式

class Local:
    def __init__(self):
        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]


"""
由于实例化时会自动调用Local中的__init__方法,执行self.storage ={}
而self.storage ={}则会自动执行__setattr__方法
会先获取ident = get_ident()然后走else分支self.storage[ident] = {k: v}!!!此时又有self.storage又会再次执行__setattr__所有会进入死循环
最终报错
"""
obj = Local()

def task(arg):
    obj.val = arg
    print(obj.storage)
    time.sleep(2)
    print(obj.val)


for i in range(10):
    t = Thread(target=task, args=(i,))
    t.start()

3.面向对象优化

class Local:
    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[k][item]


"""
由于实例化时会自动调用Local中的__init__方法,执行self.storage ={}
而self.storage ={}则会自动执行__setattr__方法
会先获取ident = get_ident()然后走else分支self.storage[ident] = {k: v}!!!此时又有self.storage又会再次执行__setattr__所有会进入死循环
最终报错
解决方法,初始化时调用父类object去创建storage的空字典
"""
obj = Local()


def task(arg):
    obj.val = arg
    print(obj.storage)
    time.sleep(2)
    print(obj.val)


for i in range(10):
    t = Thread(target=task, args=(i,))
    t.start()

优于threading.local地方

可以为每个线程(协程)开辟一块空间进行数据存储

try:
    from greenlet import getcurrent as get_ident
except Exception:

    from threading import get_ident
import time
from threading import Thread

class Local:
    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]

2.上下文管理

# 在源码中分析上下文管理
第一阶段:
	-将ctx(request,session)放到"空调"上(Local对象)

第二阶段:
	-将视图函数导入:request/session
第三阶段:
	-请求处理完毕
		-获取session并保存到cookie
    	-将ctx删除

flask 3 上下文管理、threading.local相关、基础知识

# flask中session是什么时候产生什么时候销毁的
当请求进来时,会将request和session封装成一个RequestContext对象,该对象 通过localstack放入内部的local对象中此时session是None值,接着执行open_session将cookie中值拿来放入session中
最后返回时执行save_session将ctx中的读出来进行序列化写入cookie,最后pop将ctx去除

视图阶段的上下文管理

from flask import Flask, request, session

app = Flask(__name__)


@app.route('/')
def hello_world():
    """
    request = LocalProxy(partial(_lookup_req_object, "request"))
    session = LocalProxy(partial(_lookup_req_object, "session"))
    def _lookup_req_object(name):
    # name = request/session
    # top = ctx(request,session)
    top = _request_ctx_stack.top
    if top is None:
        raise RuntimeError(_request_ctx_err_msg)
    # 从ctx中获取name对象  request/session
    return getattr(top, name)


    """
    print(request)  # LocalProxy.__str__
    print(request.method)  # LocalProxy.__getattr__(key="method")  # 先去ctx中获取request,再去request中获取method
    print(request.args)  # LocalProxy.__getattr__(key="args")  # 先去ctx中获取request,再去request中获取args

    session['k1'] = 123  # LocalProxy.__setattr__(key="k1")  # 先去ctx中获取session,再去session中给k1设置值
    print(session['k1'])  # LocalProxy.__getattr__(key="k1")  # 先去ctx中获取session,再去session中获取args

    """
    class LocalProxy(object):
        def __init__(self, local, name=None):
        # self.__local = 函数
        object.__setattr__(self, "_LocalProxy__local", local)
        object.__setattr__(self, "__name__", name)
        if callable(local) and not hasattr(local, "__release_local__"):
            # "local" is a callable that is not an instance of Local or
            # LocalManager: mark it as a wrapped function.
            object.__setattr__(self, "__wrapped__", local)
        def __getattr__(self, name):
            if name == "__members__":
                return dir(self._get_current_object())
            # name = method
            # obj = self._get_current_object()  去ctx中获取request
            # return getattr(obj, name)  request.method
            return getattr(self._get_current_object(), name)
        
        def _get_current_object(self):
            if not hasattr(self.__local, "__release_local__"):
                return self.__local()
            try:
                return getattr(self.__local, self.__name__)
            except AttributeError:
                raise RuntimeError("no object bound to %s" % self.__name__)
    """


    return 'hello world'



if __name__ == '__main__':
    app.run()



"""
用户请求一旦到来请求:
    1.app.__call__=========>def __call__(self, environ, start_response):
                                return self.wsgi_app(environ, start_response)
    其中environ是由wsgi操作完封装好的
    2.ctx = self.request_context(environ):RequestContext(self, environ)==========>就是一个团队,内有request和session
        -ctx.request = Request(environ) = request
        -ctx.session = None
        -路由匹配
"""

flask 3 上下文管理、threading.local相关、基础知识

一个请求的完整流程

flask 3 上下文管理、threading.local相关、基础知识

问题:flask*有几个LocalStack和Local对象

都是两个

g是什么

是每一次请求的全局变量,在该次请求结束后销毁,可在before_request中给g赋值,在该次请求后面的操作中可以用该变量去取值

上一篇:Python多进程及多线程基础


下一篇:线程、进程、协程