zoukankan      html  css  js  c++  java
  • Flask基础

    一,Flask初始

    Python三大Web框架对比

    1.Django 主要特点是大而全,集成了很多组件,例如: Models Admin Form 等等, 不管你用得到用不到,反正它全都有,属于全能型框架

    2.Tornado 主要特点是原生异步非阻塞,在IO密集型应用和多任务处理上占据绝对性的优势,属于专注型框架

    3.Flask 主要特点小而轻,原生组件几乎为0, 三方提供的组件请参考Django 非常全面,属于短小精悍型框架

    Django 通常用于大型Web应用由于内置组件足够强大所以使用Django开发可以一气呵成

    Tornado 通常用于API后端应用,游戏服务后台,其内部实现的异步非阻塞真是稳得一批

    Flask 通常应用于小型应用和快速构建应用,其强大的三方库,足以支撑一个大型的Web应用

    Django 优点是大而全,缺点也就暴露出来了,这么多的资源一次性全部加载,肯定会造成一部分的资源浪费

    Tornado 优点是异步,缺点是干净,连个Session都不支持

    Flask 优点是精悍简单,缺点是你不会!哈哈哈哈哈哈!

    Flask:
    1.轻,短小精悍
    2.快,三行代码开启服务
    缺点:
    1.组件大部分来源于三方,flask-admin,flask-session
    2.flask大版本更新,组件更新速度慢
        
    Django:
    1.大而全,admin,models,Form,中间件,session
    2.一个框架解决所有问题
    缺点:
    1.一旦启动,所有资源全部加载,用不到的,浪费了
    2.太大了,结构复杂
    3.所有的组件,全部由Django自身控制
    
    Tornado:
    1.原生websocket
    2.异步io
    3.非阻塞
    缺点:
    三方及原生组件几乎为0

    二. Flask 的安装与HelloWorld

    Flask的安装特别难,但是以一个多年奋斗在程序界的我,肯定会找出一个最简单的方法教你们,具体操作如下:

    pip install Flask

    别问我还有没有复杂的方法,没有!

    Flask安装完成了,下面使用Flask走一遍仪式:

    三行代码

    from flask import Flask
    app = Flask(__name__)
    app.run()

    默认端口是5000,因为没有定义路由,所以报404。但是服务是起来了!

    六行代码

    from flask import Flask  # 导入Flask类
    app = Flask(__name__)  # 实例化Flask对象app
    @app.route("/")    # app中的route装饰器
    def index():      # 视图函数
        return "HelloWorld!!"  # 返回响应体
    # 监听地址为0.0.0.0,表示服务器的所有网卡
    # 5000是监听端口
    # debug=True表示启动debug模式。当代码有改动时,Flask会自动加载,无序重启!
    app.run()  # 启动Flask服务

    注意:默认的debug模式是关闭的。如有有代码改动,需要重启flask才行生效!

    但是开启debug模式,代码一有改动,会立刻加载,无需重启!

    还有一点,app = Flask(__name__)。这里面的__name__表示 标识模块的名字的一个系统变量

    还可以是 app= Flask("fdsafejisi"),这样运行也没有问题。那么为什么要用__name__呢?

    后面学习到蓝图会用到!

    启动了Flask,得到了返回值,打印在页面上

    二、Response三剑客

    HTTPResponse

    @app.route("/")  # app中的route装饰器
    def index():  # 视图函数
        return "HelloWorld!!"  # HttpResponse

    在Flask 中的HttpResponse 在我们看来其实就是直接返回字符串

    Redirect

    from flask import Flask  # 导入Flask类
    from flask import redirect  # 导入flask中的redirect
    
    app = Flask(__name__)
    
    
    # app中的route装饰器,用来指定视图函数的URL地址
    @app.route("/redi")
    def redi():  # 视图函数
        return redirect("/")  # redirect跳转至"/"
    
    
    @app.route("/")
    def index():  # 视图函数
        return "hello"
    
    
    if __name__ == '__main__':
        app.run("0.0.0.0", 5000, debug=True)
    View Code

    每当访问"/redi"这个地址的时候,视图函数redi会触发redirect("/") 跳转到url地址:  "/" 并会触发"/"对应的视图函数index()

    render (render_template)

    from flask import Flask  # 导入Flask类
    from flask import render_template  # 导入flask中的render_template
    
    app = Flask(__name__)
    
    
    @app.route("/home")
    def home():  # 视图函数
        # 渲染html模板,返回html页面
        return render_template("home.html")
    
    
    if __name__ == '__main__':
        app.run("0.0.0.0", 5000, debug=True)

    在当前py文件目录中创建templates,在此目录下创建文件home.html

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    <h1>Flask</h1>
    </body>
    </html>

    目录结果如下:

    ./
    ├── demo.py
    └── templates
        └── home.html

    为什么要一定要创建templates文件夹呢?叫abc,行不行呢?不行!

    看这一行代码

    app = Flask(__name__)

    使用Ctrl+鼠标左键,点击Falsk,查看源码

    def __init__(
        self,
        import_name,
        static_url_path=None,
        static_folder='static',
        static_host=None,
        host_matching=False,
        subdomain_matching=False,
        template_folder='templates',
        instance_path=None,
        instance_relative_config=False,
        root_path=None
    ):

    看到template_folder变量没有?文件夹必须叫这个名字!

    三、Request(全局变量,独有机制--Flask请求上下文管理)

    每个框架中都有处理请求的机制(request),但是每个框架的处理方式和机制是不同的

    为了了解Flask的request中都有什么东西,首先我们要写一个前后端的交互

    基于HTML + Flask 写一段前后端的交互

    先写一段儿HTML form表单中提交方式是post  action地址是 /req

    在templates目录创建文件login.html

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    <h1>欢迎登陆</h1>
    <form action="/req" method="post">
        <p>
            <input type="text" name="user" placeholder="请输入用户名">
        </p>
        <p>
            <input type="password" name="pwd" placeholder="请输入密码">
        </p>
        <input type="submit" value="提交">
    </form>
    </body>
    </html>
    登录界面

    写好一个标准 form 表单,一点提交,搜就向后端提交一个POST请求过去了

    后端的接收方式就 666 了

    首先要从 flask 包中导入 request 模块 , 至于为什么要导入 request 呢? 这里不做解释,暂时你就知道 request 如果要用,需要导入

    from flask import Flask  # 导入Flask类
    from flask import render_template  # 导入flask中的render_template
    from flask import request  # 导入flask中的request
    
    app = Flask(__name__)
    
    @app.route("/login")
    def login():
        return render_template("login.html")
    
    
    @app.route("/req",methods=["POST"])  # 只允许POST
    def home():  # 视图函数
        print(request)  # request对象
        print(request.method)  # POST看来可以使用这种方式来验证请求方式
        # ImmutableMultiDict([('user', 'xiao'), ('pwd', '123')])
        print(request.form)
        # ImmutableMultiDict 它看起来像是Dict,使用字典方式取值
        print(request.form["user"])  # xiao
        print(request.form.get("pwd"))  # 123
        # 字典迭代器对象,keys表示获取所有值
        print(request.form.keys())
        # 既然是迭代器,就可以使用for循环了
        for i in request.form.keys():
            print(i)
    
        return "ok"
    
    
    if __name__ == '__main__':
        app.run("0.0.0.0", 5000, debug=True)

    解释一个 @app.route("/req",methods=["POST"]) :

    methods=["POST"]  代表这个url地址只允许 POST 请求,是个列表也就是意味着可以允许多重请求方式,例如GET之类的

    储存数据:

    1.request.args

      request.args.get("name") url中取值
      print(request.args)  # ImmutableMultiDict([('id', '1'), ('age', '20')])
      print(request.args["id"])  # 1
      print(request.args.get("age"))  # 20
      print(list(request.args.keys()))  # ['id', 'age']
      print(list(request.args.values()))  # ['1', '20']  
      req_dict = dict(request.args)  # {'id': ['1'], 'age': ['20']}
      print(req_dict)

    2.request.form

    Form表单中传递过来的值 使用 request.form 中拿到
    print(request.form)  # ImmutableMultiDict([('user', 'xiao'), ('pwd', '123')])
    # ImmutableMultiDict 它看起来像是的Dict 就用Dict的方法取值试一下吧
    print(request.form["user"])  # xiao
    print(request.form.get("pwd"))  # 123
    # 看来全部才对了, ImmutableMultiDict 似乎就是个字典,再来玩一玩它
    print(list(request.form.keys()))  # ['user', 'pwd'] 看来是又才对了
    #如果以上所有的方法你都觉得用的不爽的话
    req_dict = dict(request.form)
    print(req_dict)  # 如果你觉得用字典更爽的话,也可以转成字典操作(这里有坑)

    3.request.json

    如果在请求中写入了 "application/json" 使用 request.json 则返回json解析数据, 否则返回 None

    4.request.data 

    如果处理不了的就变成字符串儿存在data里面 

    5..request.values(禁用)

    只要是个参数我都要

     属性数据:

    1.request.method  :   验证请求方式

    2.request.url    request.path

       # 获取当前的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/

    3.request.files :  给我一个文件我帮你保管

    如果遇到文件上传的话,request.files 里面存的是你上传的文件,但是 Flask 在这个文件的操作中加了一定的封装,让操作变得极为简单

    print(request.files)  # ImmutableMultiDict([('file', <FileStorage: 'DragonFire.txt' ('text/plain')>)])
    print(request.files["file"])  # <FileStorage: 'DragonFire.txt' ('text/plain')>
    my_file = request.files["file"]
    my_file.save("OldBoyEDU.txt")  # 保存文件,里面可以写完整路径+文件名

     注意:前端页面必须设置enctype="multipart/form-data",否则提交时,会报错

     <form action="/req" method="post" enctype="multipart/form-data">

    4.request.headers  : 用来获取本次请求的请求头

    5.request.cookies : 存在浏览器端的字符串儿也会一起带过来

    前提是你要开启浏览器的 cookies

    request.cookies 是将cookies中信息读取出来

    *form表单的坑

    如果url和form中的Key重名的话,form中的同名的key中value会被url中的value覆盖
    <form action="/req?id=1&user=20" method="post"> <p> <input type="text" name="user" placeholder="请输入用户名"> </p> <p> <input type="password" name="pwd" placeholder="请输入密码"> </p> <input type="submit" value="提交"> </form>
    如果url和form中的Key重名的话,form中的同名的key中value会被url中的value覆

    四、模板语言 Jinja2

    Flask中默认的模板语言是Jinja2

    Jinja2模板语言中的 for

    {% for foo in g %}
    {{ foo }}
    {% endfor %}

    Jinja2模板语言中的 if

    {% if g %}
    
    {% elif g %}
        
    {% else %}
        
    {% endif %}

    Jinja2 的高阶用法

    <input type='text' name='user' value='xiao'>

    safe  显示渲染后的效果

    后端代码:

    from flask import Flask  # 导入Flask类
    from flask import render_template  # 导入flask中的render_template
    
    app = Flask(__name__)
    
    @app.route("/")
    def index():
        tag = "<input type='text' name='user' value='xiao'>"
        return render_template("index.html",tag=tag)
    
    if __name__ == '__main__':
        app.run("0.0.0.0", 5000, debug=True)

    前端代码:

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    
        {{ tag|safe }}
    
    </body>
    </html>

    或者

    后端代码:

    from flask import Flask  # 导入Flask类
    from flask import render_template  # 导入flask中的render_template
    from flask import Markup  # 导入 flask 中的 Markup 模块
    
    app = Flask(__name__)
    
    @app.route("/")
    def index():
        tag = "<input type='text' name='user' value='xiao'>"
        # Markup帮助咱们在HTML的标签上做了一层封装,让Jinja2模板语言知道这是一个安全的HTML标签
        markup_tag = Markup(tag)
        # <input type='text' name='user' value='DragonFire'> <class 'markupsafe.Markup'>
        print(markup_tag,type(markup_tag))
    
        return render_template("index.html", tag=markup_tag)
    
    if __name__ == '__main__':
        app.run("0.0.0.0", 5000, debug=True)

    前端代码:

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    
        {{ tag }}
    
    </body>
    </html>

    执行Python函数

    模板中执行函数,首先在文件中定义一个函数

    后端代码:

    from flask import Flask  # 导入Flask类
    from flask import render_template  # 导入flask中的render_template
    
    app = Flask(__name__)
    
    #定义一个函数,把它传递给前端
    def a_b_sum(a,b):
        return a+b
    
    @app.route("/")
    def index():
        return render_template("index.html", tag=a_b_sum)
    
    if __name__ == '__main__':
        app.run("0.0.0.0", 5000, debug=True)

    前端代码:

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    
        {{ tag }}
        <br>
        {#传入2个参数#}
        {{ tag(99,1) }}
    
    </body>
    </html>

    还可以定义全局函数,无需后端传递给前端,Jinja2直接就可以执行的函数

    后端代码:

    from flask import Flask  # 导入Flask类
    from flask import render_template  # 导入flask中的render_template
    
    app = Flask(__name__)
    
    @app.template_global()  # 定义全局模板函数
    def a_b_sum(a, b):
        return a + b
    
    
    @app.template_filter()  # 定义全局模板函数
    def a_b_c_sum(a, b, c):
        return a + b + c
    
    
    @app.route("/")
    def index():
        return render_template("index.html", tag="")
    
    if __name__ == '__main__':
        app.run("0.0.0.0", 5000, debug=True)

    前端代码:

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    
        {#函数#}
        {{ a_b_sum(99,1) }}
        <br>
        {#过滤器#}
        {{ 1 | a_b_c_sum(197,2) }}
    
    </body>
    </html>

    两个函数的调用方式不太一样

    尤其是@app.template_filter() 它的调用方式比较特别,这是两个Flask中的特殊装饰器

  • 相关阅读:
    关于在php+apache开发过程中使用svn进行版本的维护
    Fragment的切换动画实现
    IOS MJExtension json转模型的轻量级框架的使用
    Centos 配置Red5流媒体服务器
    在Centos 6.5 上面配置 SVN
    在Centos 上面配置Openfire
    关于阿里云上面的Centos上面配置 防火墙
    【Android 一些难以理解的控件、容易混淆的、多种实现方式的、一些该纠正的想法】
    【进攻移动开发_htm5_跨平台_的号角】
    【进攻Android的号角】
  • 原文地址:https://www.cnblogs.com/Mr-Murray/p/9775512.html
Copyright © 2011-2022 走看看