转:https://www.cnblogs.com/li-li/p/10252058.html#main

一、flask请求上下文源码解读

  通过上篇源码分析,我们知道了有请求发来的时候就执行了app(Flask的实例化对象)的__call__方法,而__call__方法返回了app的wsgi_app(environ, start_response)方法的执行结果,而wsgi_app方法中有这样一句话:ctx = self.request_context(environ),还分析除了ctx是RequestContext类的实例化对象,而且ctx中含有有本次请求的request对象和session对象。

  接下来我们重点分析flask是如何做到把request对象当成全局变量,而又保证了数据安全,即请求信息互不影响的。

1、flask请求上文源码解读

  上篇我们分析到了如何得到RequestContext实例化对象ctx,接下来ctx对象执行push方法,如下:

  RequestContext类中的push方法源码如下:

  _request_ctx_stack是LocalStack类的实例化对象:

  

  LocalStack类中的__init__方法如下:

  Local类的__init__方法如下:

  get_ident是Local类所在文件中导入的一个方法名,该方法执行后会得到线程或协程ID,如下:

  LocalStack类中的top是一个属性方法,源码如下:

  下一步Local类中的__getattr__方法源码如下:

  到此,分析得出top = _request_ctx_stack.top中的top为None。

  接下来分析 _request_ctx_stack.push(self)做了什么?LocalStack类中的push方法源码如下:

  Local类中的__setattr__方法源码如下:

  因为rv.append(obj),所以最后LocalStack对象,即_request_ctx_stack对象字典化后如下:

  {'_local':{'__storage__':{9527:{stack:[ctx]}}, '__ident_func__':get_ident}}
  # 说明:9527假设是获取到的线程或者协程号,ctx包含request对象和session对象。

  到此,flask请求上文结束,也就是完成了将一个request和session对象存储到某个地方。

2、下文

  我们知道flask的request对象和session对象是全局变量,上文已经解读了如何存储。接下来解读如何在保证数据安全的情况下取出来,即只取到自己的请求信息而非其他人的。

  我们还知道request对象中存储了很多信息,如request.method存储请求方式、request.json存储json标准字符串等等。下面以request.method为例,分析如何得到请求方式信息。

  导入request方式如下:

  from flask import request

  源码如下:

  LocalProxy类的__init__方法如下:

  偏函数中的原函数_lookup_req_object源码如下:

  当执行request.method的时候,执行LocalProxy的__getattr__方法,源码如下:

  查看类LocalProxy中的_get_current_object方法是如何得到本次请求的request对象,源码如下:

  至此,我们已经分析出了如何得到本次请求的request对象,从而取出request对象中的相关信息。

二、http聊天室(单聊/群聊)- 基于gevent-websocket

1、准备知识

  http协议特点:短连接,无状态保存;

  轮询:前后端一秒交互多次,压力极大,并且消耗带宽,资源浪费极其严重;

  长轮询:即让服务器保存我的一个连接状态,用于快速传递消息,节省带宽,释放压力,数据实时性强;

  长连接:服务端及客户端节省极大的资源,能保证数据实时性;

  带宽:1Mbps  = 128KB/s

2、http聊天室

  准备工作:下载gevent-websocket模块

1
pip3 install gevent-websocket

  代码示例:

  manage.py代码:

  from flask import Flask, request, render_template
  from geventwebsocket.handler import WebSocketHandler
  from geventwebsocket.websocket import WebSocket # 提示用
  from gevent.pywsgi import WSGIServer
  import json   app = Flask(__name__)   user_socket_dict = {} # 用户字典   @app.route('/ws/<username>')
  def ws(username):
    print(request.environ) # 有个wsgi.websocket,通过它可以发消息
    user_socket = request.environ.get('wsgi.websocket') #type:WebSocket
    if user_socket:
      user_socket_dict[username] = user_socket
    print(user_socket_dict)
    while 1:
      msg = user_socket.receive()
      msg_dict = json.loads(msg)
      msg_dict['from_user'] = username
      to_user = msg_dict.get('to_user')
      # chat = msg_dict.get('msg')
      u_socket = user_socket_dict.get(to_user) #type:WebSocket
      u_socket.send(json.dumps(msg_dict))   @app.route('/')
  def index():
    return render_template('ws.html')   if __name__ == '__main__':
    http_serv = WSGIServer(('0.0.0.0',9527), app, handler_class=WebSocketHandler)
    http_serv.server_forever()

  ws.html代码:

  <!DOCTYPE html>
  <html lang="en">
  <head>
    <meta charset="UTF-8">
    <title>Title</title>
  </head>
  <body>     <input id="username"type="text"><button onclick="login()">登录聊天室</button>     给<input id="to_user"type="text">     <input id="msg"type="text"><button onclick="send_msg()">发送</button>     <div id="chat_list"style="width:500px; height:500px; border:1px solid red;"></div>   </body>   <script type="text/javascript">   var ws = null; // 因其他函数也可能会用到ws,所以不能放在某一个函数中   function login() {
    var username = document.getElementById('username').value;
    var ws = new WebSocket('ws://192.168.13.172:9527/ws'+username); // ws请求协议
    ws.onmessage = function (data) {
      console.log(data.data);
      var recv_msg = JSON.parse(data.data);
      var ptag = document.createElement('p');
      ptag.innerText = recv_msg.from_user + ':' + recv_msg.msg;
      document.getElementById('caht_list').appendChild(ptag)
    };
  }   function send_msg() {
    var to_user = document.getElementById('to_user').value;
    var msg = document.getElementById('msg').value;
    var send_dict = {
      'to_user':to_user,
      'msg':msg
    };
    ws.send(JSON.stringify(send_dict));
  }   </script>   </html>

flask 请求上下文源码(转)的更多相关文章

  1. Flask(4)- flask请求上下文源码解读、http聊天室单聊/群聊(基于gevent-websocket)

    一.flask请求上下文源码解读 通过上篇源码分析,我们知道了有请求发来的时候就执行了app(Flask的实例化对象)的__call__方法,而__call__方法返回了app的wsgi_app(en ...

  2. Flask请求上下文源码讲解,简单的群聊单聊web

    请求上下文流程图 群聊html代码 <!DOCTYPE html> <html lang="en"> <head> <meta chars ...

  3. flask请求上下文源码分析

    一.什么是上下文 每一段程序都有很多外部变量,只有像add这种简单的函数才是没有外部变量的,一旦你的一段程序有了外部变量,这段程序就不完整了,不能独立运行,你为了使他们能运行,就要给所有的外部变量一个 ...

  4. flask的请求上下文源码解读

    一.flask请求上下文源码解读 通过上篇源码分析( ---Flask中的CBV和上下文管理--- ),我们知道了有请求发来的时候就执行了app(Flask的实例化对象)的__call__方法,而__ ...

  5. Flask系列10-- Flask请求上下文源码分析

    总览 一.基础准备. 1. local类 对于一个类,实例化得到它的对象后,如果开启多个线程对它的属性进行操作,会发现数据时不安全的 import time from threading import ...

  6. Flask框架 (四)—— 请求上下文源码分析、g对象、第三方插件(flask_session、flask_script、wtforms)、信号

    Flask框架 (四)—— 请求上下文源码分析.g对象.第三方插件(flask_session.flask_script.wtforms).信号 目录 请求上下文源码分析.g对象.第三方插件(flas ...

  7. Flask的上下文源码剖析

    先写一段Flask程序 from flask import Flask app = Flask(__name__) @app.route('/') def hello_world(): return ...

  8. Flask请求和应用上下文源码分析

      flask的request和session设置方式比较新颖,如果没有这种方式,那么就只能通过参数的传递. flask是如何做的呢? 1:本地线程,保证即使是多个线程,自己的值也是互相隔离 1 im ...

  9. Flask之 请求,应用 上下文源码解析

    什么是上下文? 每一段程序都有很多外部变量.只有像Add这种简单的函数才是没有外部变量的.一旦你的一段程序有了外部变量,这段程序就不完整,不能独立运行.你为了使他们运行,就要给所有的外部变量一个一个写 ...

随机推荐

  1. style的继承

    第一种方式:瞄准控件的基类 如下例所示,继承ContentControl的控件,都可以使用这个Style <Window.Resources> <Style x:Key=" ...

  2. 基于IdentityServer4的单点登录——Client

    以MvcClient项目为例 1.新建项目并添加引用 新建一个asp .net core 2.0的项目引用IdentityModel 2.配置 比之前的控制台客户端多这个步骤,需要配置这个客户端的Cl ...

  3. 微服务架构之「 下一代微服务 Service Mesh 」

    Service Mesh 被大家称为下一代的微服务,是微服务领域的一颗新星,被大家讨论的非常多. 我在大家的讨论中,还看到有人说 “目前的微服务架构我都没学会呢,现在又来一个下一代微服务,真学不动了” ...

  4. Delphi中close与Terminate方法的区别

    在有多个Form窗体时可以体现出来.用close是只关闭本窗体,而用Application.terminate是关闭整个程序,包括所有窗体.(1)当Close是一个主窗体时,程序会退出.Close会发 ...

  5. Linux杂谈: 树形显示多级目录--tree

    最近写博客的时候偶尔会需要将文件目录结构直观地列出来,例如python的包结构. 于是在网上搜了搜,发现了一个Linux下还不错的工具--tree tree 可以很直观地显示多级目录结构. 1. 安装 ...

  6. SQL之连接查询

    这几天忙着笔试.面试,遇到了好几个关于数据库的试题,当然只是简单的多表查询.我第一时间都是选择select...from...where...group by...having...这个结构去写的.但 ...

  7. Android签名打包

    生成正式的签名APK文件 1.使用AndroidStudio生成: 点击导航栏上的Build-->Generate Signed APK,弹出创建签名APK对话框(首次点击可能会提示输入操作系统 ...

  8. FMX 动态创建 和 销毁(释放free) 对象

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 3 ...

  9. 使用MinGW编译Boost

    1.下载Boost(http://www.boost.org) 我目前用的是1.61.0版本 2.将MinGW下的bin目录完整路径设置到系统环境变量Path中,保证cmd命令行能找到gcc,g++等 ...

  10. CWnd和HWND的区别(hWnd只是CWnd对象的一个成员变量,代表与这个对象绑定的窗口)

            所有控件类都是CWnd类的派生类,CWnd的所有成员函数在控件类中都可以使用.在MFC中,CWnd类是一个很重要的类,它封装了Windows的窗口句柄HWND.在Windows编程中, ...