Flask上下文管理源码分析
上下文管理本质(类似于threading.local):
1.每一个线程都会在Local类中创建一条数据:
{
"唯一标识":{stark:[ctx,]},
"唯一标识":{stark:[ctx,]},
}
2.当请求进来之后,将请求相关数据添加到列表里面[request,],以后如果使用时,就去读取。
3.列表中的数据,请求完成之后,将request从列表中移除,
在源码中分析上下文管理:
1,存数据,2,执行视图函数,3,请求结束。
第一阶段:执行__call__--->app.wsgi-->将ctx(request,session)封装为RequestContext()在(open_session),app_ctx(g,app)封装为APPContent()通过LocalStack将这两个类放入Local对象中。
# 在封装到RequestContext()时,
# request = Request对象
# session = None # 在执行ctx.push()时,才给session赋予了值。 # ctx.session = self.session_interface.open_session(self,request)
第二阶段:视图函数导入:request/session/g/app,通过偏函数(_lookup_req_object)在通过(LocalProxy())去LocalStack中的Local类中对其进行增删改查操作。
# 执行视图函数 # response = self.full_dispathch_request() 读取ctx.session将数据写入到浏览器的cookie中。 # 执行偏函数 # request = LocalProxy(partial(_lookup_req_object,"reqeust"))
# _local = 偏函数
# __getattr__
# ctx.request.method ( top, pop ....) # ctx.request.args # session 也在此时写入数据 # session = LocalProxy(partial(_look_req_object,"session")) # _local = 偏函数
# __setitem__
# ctx.session["xx"] == 123
第三阶段:请求处理完毕:
通过save_session将签名session保存到cookie.
通过ctx.pop()去LocalStack中的Local类中将ctx删除。
有关面试问题:
flask和django的区别:
对于django来说,内部组件特别多,自身功能强大,大而全,。
flask内置组件很少,但是第三方组件很多,扩展性强。
因为他们两个框架都没有写socket,都是基于wsgi协议做的,除此之外,flask框架中的上下文管理较为耀眼。
相同点:都没有socket,都是基于wsig协议做的。
不同点:
请求相关的数据传递的方式不同:
django:通过传递request参数取值。
f'lask:
组件不同:
django:大而全。
flask:第三方组件多。
flask上下文管理:
简单来说,flask上下文管理可以分为三个阶段:
1,请求进来时,将请求相关的数据放入上下文管理中。
2,在视图函数中,要去上下文管理中取值。
3,请求响应,要将上下文管理中的数据清除。
详细点来说:
1,请求刚进来,将request,session封装在RequestContext类中,app,g封装在AppContext类中,并通过LocalStack将reqeustcontext和appcontext放入Local类中。
2,视图函数中,通过LocalProxy-->偏函数-->LocalStack-->Local取值。
3,请求响应时,先执行save.session()再各自执行pop(),将Local中的数据清除。
Flask中的session是什么时候创建,什么时候销毁:
当请求进来时,会将request和session封装为一个RequestContext对象。通过LocalStack将RequestContex放入到Local对象中,因为请求第一次进来的时候,session是空值,所以执行open_session,给session赋值,再通过视图函数处理,请求响应时执行save.session,将签名session写入到用户浏览器的cookie中,再将Local中的数值pop掉。
Flask中一共有几个LocalStack和Local对象。
有两个LocalStack,两个Local。
request和session共用一个LocalStack和Local。
g和 app 共用一个LocalStack 和 Local.
为什么把请求放到RequestContext中:
因为request和session都是在视图中操作频繁的数据,也是用户请求需要用的数据。将request和session封装在RequestContext中,使用top,pop方法,一次就可以完成。如果单独封装,就会操作频繁。
ctx = RequestContext(request,session)
Local的作用:
保存数据,请求上下文对象和app上下文对象。
LocalStack的作用:
将Local对象中的数据维护成一个栈 [ctx,ctx,....](先进后出)
{ "协程或线程的唯一标识":{stack: [ ctx,ctx,... ] } }
为什么要维护成一个栈:
当时web应用时:不过是单线程还是多线程,栈中只有一个数据。
服务端单线程:{111:{stack:[ ctx, ] }
服务端多线程:{111:{stack:[ ctx, ] },112:{stack: [ctx, ] }}
离线脚本:可以在栈中放入多个数据:
with app01.app_context():
print(current_app)
with app02.app_context():
print(current_app)
print(current_app)
什么是 g:
g 相当于一次请求的全局变量,当请求进来时将g和current_app封装为一个APPContext类,再通过LocalStack将AppContext放入到Local类中,取值时t通过偏函数,LocalStack,Local中取值,响应时将Local中的数据g删除。
上下文管理的技术点:
反射:LocalProxy()
面向对象,封装:RequestContext
线程。
Flask上下文管理源码分析的更多相关文章
- Flask上下文管理源码分析 ——(3)
引出的问题 Flask如何使用上下文临时把某些对象变为全局可访问 首先我们做如下的几种情况的假设 情况一:单进程单线程 这种情况可以基于全局变量存储临时的对象 情况二:单进程多线程 这种情况会出现多个 ...
- Flask上下文管理源码--亲自解析一下
前戏 偏函数 def index(a,b): return a+b # 原来的调用方法 # ret=index(1,2) # print(ret) # 偏函数--帮助开发者自动传递参数 import ...
- Flask系列之源码分析(一)
目录: 涉及知识点 Flask框架原理 简单示例 路由系统原理源码分析 请求流程简单源码分析 响应流程简单源码分析 session简单源码分析 涉及知识点 1.装饰器 闭包思想 def wapper( ...
- java 1.8 动态代理源码分析
JDK8动态代理源码分析 动态代理的基本使用就不详细介绍了: 例子: class proxyed implements pro{ @Override public void text() { Syst ...
- Spring Boot自动装配原理源码分析
1.环境准备 使用IDEA Spring Initializr快速创建一个Spring Boot项目 添加一个Controller类 @RestController public class Hell ...
- Flask Session 使用和源码分析 —— (6)
基本使用 from flask import Flask, session, redirect, url_for, escape, request app = Flask(__name__) @app ...
- flask/app.py-add_url_rule源码分析
之前分析route方法的时候,可以看到中间会调用add_url_rule方法,add_url_rule方法和route方法一样属于Flask这个类的. add_url_rule方法主要用来连接url规 ...
- JDK7动态代理源码分析
IObject proxy = (IObject) Proxy.newProxyInstance(IObject.class.getClassLoader(), new Class[]{IObject ...
- (转)Spring对注解(Annotation)处理源码分析1——扫描和读取Bean定义
1.从Spring2.0以后的版本中,Spring也引入了基于注解(Annotation)方式的配置,注解(Annotation)是JDK1.5中引入的一个新特性,用于简化Bean的配置,某些场合可以 ...
随机推荐
- Codeforces 1096G. Lucky Tickets【生成函数】
LINK 题目大意 很简单自己看 思路 考虑生成函数(为啥tags里面有一个dp啊) 显然,每一个指数上是否有系数是由数集中是否有这个数决定的 有的话就是1没有就是0 然后求出这个生成函数的\(\fr ...
- list.stream().parallel() 并行流
https://blog.csdn.net/u011001723/article/details/52794455/ : parallel()其实就是一个并行执行的流.它通过默认的ForkJoin ...
- java8 流操作 好文网址
api 各方法详解(很不错!) http://blog.51cto.com/turnsole/2093185 api 各方法 简介: https://www.cnblogs.com/guguli/p/ ...
- python list 去掉重复元素
貌似用遍历最方便. http://www.cnblogs.com/tudas/p/python-delete-duplicate-element-from-list.html
- Hadoop storm大数据分析 知识体系结构
最近工作工作有用到hadoop 和storm,最近看到一个网站上例句的hadoop 和storm的知识体系.所以列出来供大家了解和学习.来自哪个网站就不写了以免以为我做广告额. 目录结构知识点还是挺全 ...
- 使用cglib动态创建javabean
动态创建javabean对于我们进行开发,会有一定的帮助,一下是使用cglib动态创建javabean的一个例子: 先贴上code: package com.dalong.CreateCode; im ...
- C#处理Exception的常用方法总结
在.NET中,异常是指成员没有完成它的名称宣称可以完成的行动.在异常的机制中,异常和某件事情的发生频率无关. 异常处理四要素包括:一个表示异常详细信息的类类型:一个向调用者引发异常类实例的成员:调用 ...
- 如何开启GZIP
服务器设置 gzip 压缩是 web 开发里很普遍的做法.假设你要请求一个 100k 的文件,网络传输速度为 50k/s,需要 2s 才能得到数据,但是如果在服务器设置了 gzip 压缩,将服务端的文 ...
- [脚本] 一个用于BMP到EPS转换的BAT脚本实现(需要安装bmeps)
最近用LaTeX写文章, 图片需要使用eps格式. 如果你安装了bmeps这个工具(一般你装了CTeX就自带这个工具的), 可以在需要转换的目录打开CMD窗口, 然后输入: bmeps -c a.jp ...
- C#如何使用VS2010与SQL2008建立链接及初步调用(转)
关于VS2010与SQL2008建立链接及初步调用问题,网上参考的资料很多,我写这个博客,并非是做重复工作,也不是做搬运工.本文将以一种初学者的角度,去完成从数据库建立,到VS2010与SQL中的数据 ...