一、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聊天室(单聊/群聊)

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&&人工智能AI --4的更多相关文章

  1. Flask&&人工智能AI -- 6 人工智能初识,百度AI,图灵机器人

    一.人工智能初识,百度AI的创建与应用 参考连接:https://www.cnblogs.com/xiao987334176/p/9620113.html 1. 百度ai开放平台 2. AipSpee ...

  2. Flask&&人工智能AI -- 8 HTML5+ 初识,HBuilder,夜神模拟器,Webview

    昨日内容回顾 1.增删改查: 增: db.collections.insert({a:1}) // 官方不推荐了 db.collections.insertMany([{a:1},{b:1}]) in ...

  3. Flask&&人工智能AI --1

    Flask初识,Response三剑客,jsonify以及send_file.Request,模板语言 Jinja2,用户登录例子,内置Sessio 一.Flask初识 首先,要看你学没学过Djang ...

  4. Flask&&人工智能AI --2

    参考博客: https://www.cnblogs.com/xiao987334176/p/9598606.html 昨日作业讲解 昨天的作业就是,有3个视图函数,分别是/login,/student ...

  5. Flask&&人工智能AI --3

    一.flask中的CBV 对比django中的CBV,我们来看一下flask中的CBV怎么实现? from flask import Flask, render_template, url_for, ...

  6. Flask&&人工智能AI -- 7 MongoDB

    MongoDB初识,增删改查操作,数据类型,$关键字以及$修改器.“$”的奇妙用法,Array Object的特殊操作,选取跳过排序,客户端操作 一.MongoDB初识 什么是MongoDB Mong ...

  7. Flask&&人工智能AI --5 Flask-session、WTForms、数据库连接池、Websocket

    未完待续.... DButils 什么是数据库连接池 数据库连接池负责分配.管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个:释放空闲时间超过最大空闲时间的数据库 ...

  8. 数据挖掘(data mining),机器学习(machine learning),和人工智能(AI)的区别是什么? 数据科学(data science)和商业分析(business analytics)之间有什么关系?

    本来我以为不需要解释这个问题的,到底数据挖掘(data mining),机器学习(machine learning),和人工智能(AI)有什么区别,但是前几天因为有个学弟问我,我想了想发现我竟然也回答 ...

  9. 【转】人工智能(AI)资料大全

    这里收集的是关于人工智能(AI)的教程.书籍.视频演讲和论文. 欢迎提供更多的信息. 在线教程 麻省理工学院人工智能视频教程 – 麻省理工人工智能课程 人工智能入门 – 人工智能基础学习.Peter ...

随机推荐

  1. 每天一个linux命令(12):nl命令

    版权声明更新:2017-05-16博主:LuckyAlan联系:liuwenvip163@163.com声明:吃水不忘挖井人,转载请注明出处! 1 文章介绍 本文介绍了Linux下面的mv命令. 2. ...

  2. 错误名称:Uncaught SyntaxError: Unexpected identifier

    控制台输出: 1.谷歌:Uncaught SyntaxError: Unexpected identifier 2.火狐:SyntaxError: missing ] after element li ...

  3. 51nod 1149 Pi的递推式 组合数

    题目大意: \(F(x) = 1 (0 \leq x < 4)\) \(F(x) = F(x-1) + F(x-\pi) (4 \leq x)\) 给定\(n\),求\(F(n)\) 题解: 我 ...

  4. bzoj 4540: [Hnoi2016]序列 莫队

    题目: 给定长度为n的序列:a1,a2,-,an,记为a[1:n].类似地,a[l:r](1≤l≤r≤N)是指序列:al,al+1,-,ar- 1,ar.若1≤l≤s≤t≤r≤n,则称a[s:t]是a ...

  5. tiny4412sdk-1506原生uboot卡死

    于16年2月多购买了tiny4412sdk-1506,用友善之臂(以下简称友善)的superboot是可以进入linux,而用三星原始的uboot_tiny4412-20130729则不可以.出现现象 ...

  6. HDOJ1242(延时迷宫BFS+优先队列)

    Rescue Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Subm ...

  7. UML核心元素--分析类

    分析类共有三个:边界类(boundary).控制类(control)和实体类(entity),这些分析类都是类的版型.分析类是跨越需求到设计实现的桥梁. 边界类:从需求向现实的转换过程中,任何两个有交 ...

  8. ASP.NET MVC 3:缓存功能的设计问题

    今天这一篇文章我来谈一谈在MVC 3项目中的缓存功能,以及针对缓存的一些设计上的考量,给大家参考参考. 为什么需要讨论缓存?缓存是一个中大型系统所必须考虑的问题.为了避免每次请求都去访问后台的资源(例 ...

  9. 谁是Docker的开发者

    由CHRIS DAWSON发表在thenewstack/DATA RESEARCH qianhen123/CHB译 我们分析了Docker的容器库并提出两个问题: 1.Docker的贡献者们感兴趣的其 ...

  10. macos下清除dnscache

    sudo killall -HUP mDNSResponder 参见链接