• Flask(1)- 主流web框架、初识flask


    一、Python 现阶段三大主流Web框架 Django、Tornado、Flask 对比

      Django 主要特点是大而全,集成了很多组件(例如Models、Admin、Form等等), 不管你用得到用不到,反正它全都有,属于全能型框架,通常用于大型Web应用,由于内置组件足够强大所以使用Django开发可以一气呵成,优点是大而全,缺点也就暴露出来了,这么多的资源一次性全部加载,肯定会造成一部分的资源浪费;

      Tornado 主要特点是原生异步非阻塞,在IO密集型应用和多任务处理上占据绝对性的优势,属于专注型框架,通常用于API后端应用,游戏服务后台,其内部实现的异步非阻塞真是稳得一批,优点是异步,缺点是干净,连个Session都不支持;

      Flask 主要特点小而轻,原生组件几乎为0,三方提供的组件(请参考Django)非常全面,属于短小精悍型框架,通常应用于小型应用和快速构建应用,其强大的三方库,足以支撑一个大型的Web应用,优点是精悍简单,缺点是稳定性较差,主要因为其组件都是第三方提供,要等第三方收到更新版本通知后才能更新,否则会出现一些兼容问题; 

    二、初识flask

    1、安装flask

    pip install flask

    2、六行代码写出一个页面

      from flask import Flask       # 导入Flask类
    
      app = Flask(__name__)        # 实例化Flask对象app
    
      @app.route("/")              # app中的route装饰器
      def index():               # 视图函数
        return "hello world"
    
      app.run()             # 启动Flask web服务

      写完后启动项目,控制台如下图:

     

      使用浏览器访问上面地址,效果如下图:

      至此,你已经实现了flask的第一个程序!

    2、Flask中的Response"三剑客"

      我们知道django中的Response有3种形式(HttpResponse、redirect、render),对比django,来看一下flask中有哪些返回形式。

      1)返回HttpResponse对象

      @app.route("/home")
      def home():
        return "hello world"           # 相当于django中return HttpResponse("")

      2)重定向(redirect)

      from flask import redirect          # 导入flask中的redirect
      @app.route(
    "/home")   def home():     return redirect("/login") # 重定向至"/login"路径

        当访问"/home"这个路径的时候,视图函数home会重定向到路径"/login" 并会触发"/login"对应的视图函数。

      3)返回模板页面(render_template)

        from flask import render_template        # 导入flask中的render_template
    
        @app.route("/home")
        def home():
            return render_template("home.html")        # 渲染模板home.html并返回

        Flask中的render_template相当于django中的render

        注意:如果要使用render_template 返回渲染的模板,请在项目的主目录中加入一个目录 templates,如下图:

        否则会遇到jinja2的异常:

     

    3、flask中的"小儿子"

      1)返回标准的json字符串

        from flask import jsonify
    
        @app.route("/json")
        def jsons():
            d = {"name":"jinjiaodawangba"}
            return jsonify(d)

        返回json字符串,并且会在响应头中加Content-Type:application/json,即告诉浏览器数据是json字符串,浏览器收到后会自动进行反序列化,而使用json.dumps()则不会加此响应头。

      2)打开文件并返回文件内容(自动识别文件格式)

      from flask importsend_file
      @app.route(
    "/file")   def file():     return send_file("01.mp4")

        自动识别文件类型,即在返回文件内容时加一个响应头Content-Type:文件类型。

    4、Flask中的request(公共变量)

      每个框架中都有处理请求的机制,但是每个框架的处理方式和机制是不同的,为了了解Flask的request中都有什么,我们先来写一个基于html+flask前后端交互的示例。

      html页面代码如下:

      <body>
      <form action="/login"method="post">
        用户名:<input type="text"name="username">
        密码:<input type="password"name="pwd">
        <input type="submit"value="登录">
      </form>
      </body>

      flask代码如下:

      from flask import Flask, request, render_template
    
      app = Flask(__name__)
    
      @app.route("/login", methods=["POST"])
      def login():
        print(request.method)   # POST
        print(request.form)    
        # ImmutableMultiDict([('username', 'tom'), ('pwd', '123')])
        print(request.form.get("username"))   # tom
        print(request.form["pwd"])     # 123
        print(request.form.keys())    
        # <dict_keyiterator object at 0x00000215775C0138>
        print(list(request.form.keys()))  # ["username", "pwd"]
        return "OK"
      
      app.run(debug=True)

        注意:路由中的methods=["POST"]表示该url地址只允许POST请求,是个列表说明可以允许多种请求方式。

      1)客户端提交过来FormData数据在request.form中

        上面示例中我们可以看出,request.form是一个类似字典的数据,可以使用字典的取值方式得到值,并且还可以将类似字典的ImmutableMultiDict的数据用to_dict()方法转换成字典类型。

        print(request.form.to_dict()) # {'username': 'tom', 'pwd': '123'}
        print(dict(request.form))# {'username': ['TOM'], 'pwd': ['123']}

      2)request.method(保存请求方式)

      3)request.args(保存的是url中传递的参数)

      4)request.json

        当请求头中含有Content-Type:application/json时,数据会在request.json中。

      5)request.data

        如果提交时请求头中的Content-Type 无法被识别,将请求体中的原始数据保存,bytes类型。

      6)request.files(序列化文件,存储用save()方法,且可以通过filename获取文件名)

        my_file = request.files.get("my_file")
        my_file.save(my_file.filename)

      7)request.values(只要是个参数都保存在其中,用于查看,不要使用to_dict(),键重复会覆盖)

      8)request.cookies(存在浏览器端的字符串也会一起带过来)

      9)request.headres(请求头中的信息)

      10)request.获取各种路径 之 这些方法没必要记,但是要知道它存在

        # 获取当前的url路径
        print(request.path)              # /req
        # 当前url路径的上一级路径     print(request.script_root)
        
    # 当前url的全部路径     print(request.url) # http://127.0.0.1:5000/req

        # 当前url的路径的上一级全部路径     print(request.url_root ) # http://127.0.0.1:5000/

    5、jinja2

      同django的模板语法类似,flask使用的jinja2的语法,且同django的模板语法很类似,下面简单介绍一下使用方法,对比django的理解和记忆:

      首先后端有如下数据:

      STUDENT = {
        'id':1, 'name': 'zhangsan', 'age':15, 'gender': 'male'
      }
    
      STU_LIST = [
          {'id':1, 'name': 'zhangsan', 'age':15, 'gender': 'male'},
          {'id':2, 'name': 'lisi', 'age':18, 'gender': 'female'},
          {'id':3, 'name': 'wangwu', 'age':25, 'gender': 'buzhidao'}
      ]
    
      STU_DICT = {
        1: {'name': 'zhangsan', 'age':15, 'gender': 'male'},
        2: {'name': 'lisi', 'age':18, 'gender': 'female'},
        3: {'name': 'wangwu', 'age':25, 'gender': 'buzhidao'}
      }

      使用render_template传到前端模板页面index.html:

      @app.route('/index')
      def index():
        return render_template('index.html', stu=STUDENT, stu_list=STU_LIST, stu_dict=STU_DICT)

      index.html中jinja2的渲染语法如下:

      <body>
      <table border="1"cellpadding="0"cellspacing="0">
        <tr>
          <td>{{ stu.id }}</td>
          <td>{{ stu.name }}</td>
          <td>{{ stu.get('age') }}</td>
          <td>{{ stu['gender'] }}</td>
        </tr>
      </table>
      <br>
    
      <table border="1"cellpadding="0"cellspacing="0">
        {% for item in stu_list %}
        <tr>
          <td>{{ item.name }}</td>
          <td>{{ item.get('age') }}</td>
          <td>
              {% if item['gender']=='male' or item['gender']=='female' %}
                {{ item['gender'] }}
              {% else %}
                male
              {% endif %}
          </td>
        </tr>
         {% endfor %}
      </table>
      <br>
      <table border="1"cellpadding="0"cellspacing="0">    {% for key,value in stu_dict.items() %}     <tr>       <td>{{ key }}</td>       <td>{{ value.name }}</td>       <td>{{ value.get('age') }}</td>       <td>{{ value['gender'] }}</td>     </tr>    {% endfor %}   </table>   </body>

      总结:对比django的发现有几点不同,第一,也可以调用方法,但调用方法要加括号;第二,字典取值除了可以使用点语法,还有字典的get方法和[key]取值。

      渲染效果入下:

      1)引用变量和执行函数都使用 {{}}

        注意:除了传递类似上述示例的变量,还可以传递函数,如下:

        def add(a,b):
          return a+b
        @app.route(
    '/index')     def index():       return render_template('index.html', func=add)

        在模板文件index.html中直接使用{{ func(1,2) }}便可以渲染出函数执行结果。

      2)写逻辑代码时使用 {%%}

      3)Markup 安全标签字符串,相当于django中的过滤器safe,使用方法如下:

      from flask import Flask, render_template, Markup
    
      tag = Markup('<h1>这是一个h1标题</h1>')
    
      @app.route('/index')
      def index():
        return render_template('index.html', tag=tag)

      4)装饰器 @app.template_global()

        虽然我们知道可以向模板页面中传递函数,但是假如我们有很多页面,且都需要执行一个相同函数,此时我们要在模板页面使用的话就需要向每一个页面都传递,然而,使用装饰器@app.template_global()则不需要传递,可以直接在模板页面中使用,如下代码:

      @app.template_global()
      def add(a,b):
        return a+b

         在index.html中直接使用{{ add(1,2) }}即可得到执行结果。

      5)装饰器 @app.template_filter()

      @app.template_filter()
      def fil(a,b,c):
        return a+b+c

         在index.html中使用方法是{{ 9 | fil(3,4) }},跟django中的过滤器很相似,即管道符前的数作为fil函数的第一个参数,管道符后的函数调用中的函数作为后面的参数一次传递fil。

      6)jinja2中虽然有,但平常很少用到,就是“宏”

      {% macro create_input(na,ty) %}
        {{ na }} : <input type="{{ ty }}"name="{{ na }}">   # 定义函数
      {% endmacro %}
      {{ create_input(
    "username","text") }} # 调用函数

      7)继承和导入,同django的模板语法类似,flask的jinja2也可以继承和导入

        一个页面中导入另一个页面使用 {% include "header.html" %}

        在一个页面中继承另一个页面使用 {% extends "base.html" %}

        而且可以使用定义 {% block content %} 重写代码

    6、 Flask 中的session(公共变量)

      from flask import session
      app.secret_key
    = "加密字符串" # 用于序列化和反序列化 session信息

      我们知道cookie和session的区别是cookie存在客户端,session存在服务器端,但是flask中为了节省开销,Flask中默认Session 存放位置是客户端的Cookies中,因此在flask中用Session需要加密,也就是一定要加secret_key。

      添加一个session语法:session["user"] = "sfadgsdf"

      工作机制:由secret_key + session 加密后存放在浏览器的cookie中。

      验证sessions机制:当请求进入视图函数,带上cookie,将Session从cookie序列化出来,通过secret_key反序列化成字典。

      注意:当存入session时的key不一样,value一样时,则只序列化key,value指向同一个再序列化。

  • 相关阅读:
    思源黑体、思源宋体的 TTF 版本
    MySQL Workbench导入/导出SQL文件的方法
    高精度计算器
    树的直径和重心
    linux 分区 和 硬盘分区的对应关系
    layer 子页面传值给父页面
    php复制文件夹所有文件
    php删除文件夹下所有的文件夹和文件
    PHP将多级目录打包成zip文件
    云原生应用开发“12Factors”
  • 原文地址:https://www.cnblogs.com/li-li/p/10235680.html
Copyright © 2020-2023  润新知