zoukankan      html  css  js  c++  java
  • Flask

    Flask

    django和flask的区别.......

    django是个大而全的框架,flask是一个轻量级的框架。
    
    django内部为我们提供了非常多的组件:orm / session / cookie / admin / form / modelform / 路由 / 视图 / 模板 /  中间件 / 分页 / auth / contenttype  / 缓存 / 信号 / 多数据库连接 
    
    flask框架本身没有太多的功能:路由/视图/模板(jinja2)/session/中间件 ,第三方组件非常齐全。 
    注意:django的请求处理是逐一封装和传递; flask的请求是利用上下文管理来实现的。 
    

    安装flask

    pip install flask
    

    实现最简单的hello world

    from flask import Flask,request,render_template,redirect
    app = Flask(__name__) # 这里以接受包或者模块为参数,一般传递__name__  这里定义了变量app赋值为Flask()
    @app.route('/index')  # 这里相当于定义了一个路由,这个url和下面的函数存在了一个对应关系,映射关系
    def index():
        return 'hello world'
    if __name__ == '__main__':
        app.run()
    

    上段代码做了什么?

    1.首先,导入了 Flask 类。这个类的实例将会是我们的 WSGI 应用程序。
    
    2.接下来,创建一个该类的实例,第一个参数是应用模块或者包的名称。 如果你使用单一的模块(如本例),你应该使用 __name__ ,因为模块的名称将会因其作为单独应用启动还是作为模块导入而有不同( 也即是 '__main__' 或实际的导入名)。这是必须的,这样 Flask 才知道到哪去找模板、静态文件等等。详情见 Flask的文档。
    
    3.然后,我用 route() 装饰器告诉 Flask 什么样的URL 能触发我们的函数。
    
    4.这个函数的名字也在生成 URL 时被特定的函数采用,这个函数返回想要显示在用户浏览器中的信息。
    
    5.最后用 run() 函数来让应用运行在本地服务器上。 其中 if __name__ =='__main__': 确保服务器只会在该脚本被 Python 解释器直接执行的时候才会运行,而不是作为模块导入的时候。
    

    写flask步骤

    from flask import Flask,request,render_template,redirect,jsonify
    
    app = Flask(__name__) #创建flask对象,模板的默认存放在templates里,通过参数template_folder='模板文件夹名'
    
    @app.route('/index')  #路由和视图放一起,路由和函数存在映射关系
    def index():
        return 'hello world'
    	# return jsonify({'xx':'xx'})  json格式
    
    if __name__ == '__main__':
        app.run()  # 在内部调用werkzeug,让程序执行,用户请求一旦到来,就执行路由和视图
    

    实现用户登录

    from flask import Flask,request,render_template,redirect,url_for(url别名)
    
    app = Flask(__name__)
    
    @app.route('/login',methods=['GET','POST'],endpoint='login') 
    #请求方式,可以使GET,POST,当在重定向就可以  redirect(url_for('login'))
    def login():
        if request.method == 'GET': 
            pk = request.args.get('pk') #GET 获取url的值
            return render_template('login.html')
        user = request.form.get('user') #POST 获取表单提交过来的额数据
        pwd = request.form.get('pwd')
        if user == 'tj' and pwd == 'tj':
            return redirect('/index')
        else:
            error = '用户名或密码错误'
            return render_template('login.html',error=error)
    		return render_template('login.html',**{'error':error})
    @article_app.route('/article/del/<int:pk>',methods=['GET','POST'])  
    #<pk>  pk默认字符串形式,可以通过int转换,类似于django的通过正则拼接的路径
    
    
    #前端中url <a href='/del/{{pk}}'>删除</a>
    def remove(pk): 
        # pk = request.args.get('pk')
        sql('delete from article where id=%s'%pk)
        return redirect('/article/list')
    if __name__ == '__main__':
        app.run()
    

    flask蓝图(blueprint)

    什么是蓝图?

    一个蓝图定义了可用于单个应用的视图,模板,静态文件等等的集合,可以简单理解为路由分发
    

    构建蓝图

    目录结构:
    login_CURD
    	login_CURD
    		static    #存放一些配置文件
    		templates #放HTML
    		views
    			视图1
    			视图2
    			...
    		__init__.py
    	manage.py
    

    auth_app

    from flask import Blueprint,render_template,request,redirect,session
    from bms_flask.static.sql_search import sql
    import functools
    auth_app = Blueprint('auth',__name__)
    
    @auth_app.route('/login/',methods=['GET','POST'])
    def login():
        if request.method == 'GET':
            return render_template('login.html')
        else:
            user = request.form.get('name')
            pwd = request.form.get('pwd')
            sql_info = 'SELECT username,password FROM userinfo'
            db_info = sql(sql_info)
            for i in db_info:
                if user == i[0] and pwd == i[1]:
                    session['xxx'] = user #设置session
                    return redirect('/article/list')
                else:
                    return redirect('/login/')
    

    article_app

    from flask import Blueprint,render_template,request,redirect,session
    from bms_flask.static.sql_search import sql
    from bms_flask.views.login import auth
    article_app = Blueprint('article',__name__)
    
    @article_app.route('/article/list')
    def article():
        sql_info = 'SELECT * FROM article'
        db_info = sql(sql_info)
        return render_template('article.html',data_dict=db_info)
    

    init.py

    from flask import Flask
    from bms_flask.views.login import auth_app
    from bms_flask.views.article import article_app
    
    def create_app():
        app = Flask(__name__)
        app.secret_key = 'nvfjdvnfdjvk'  #session 的秘钥
        app.register_blueprint(auth_app) #注册的蓝图,url_prefix是一个参数,是url的前缀
        #app.register_blueprint(auth_app,url_prefix='/web')
        #在访问的时候url就为 xxx.xx/web/xx
        app.register_blueprint(article_app) #注册的蓝图
    
        return app
    

    flask与装饰器

    例如程序中设置了session,给需要认证的视图加认证判断,通过装饰器来写.
    
    # 装饰器
    from flask import Blueprint,render_template,request,redirect,session
    from bms_flask.static.sql_search import sql
    import functools
    auth_app = Blueprint('auth',__name__)
    
    def auth(func):
        @functools.wraps(func)  # 必须写,这时候执行__name__,结果就是被装饰的函数名,而不是inner,如果不加funtools,__name__就是inner,以后多个装饰器会造成url别名重复
        def inner(*args,**kwargs):
            user = session.get('xxx')
            if not user:
                return redirect('/login/')
            return func(*args,**kwargs)
        return inner
    
    
    # 加认证的视图
    from flask import Blueprint,render_template,request,redirect,session
    from bms_flask.static.sql_search import sql
    from bms_flask.views.login import auth
    article_app = Blueprint('article',__name__)
    
    @article_app.route('/article/del/',methods=['GET','POST'])
    @auth   # 一定要写在路由下边
    def remove():
        pk = request.args.get('pk')
        sql('delete from article where id=%s'%pk)
        return redirect('/article/list')
    
    

    flask的session(保持会话)

    flask的session是基于cookie的会话保持
    session保持在浏览器
    from flask import session #导入session
    
    from flask import Flask
    def create_app():
        app = Flask(__name__)
        app.secret_key = 'nvfjdvnfdjvk'  # session 的秘钥,不写会报错,是一个随机字符串
        return app
    
    session['xxx'] = user  # 设置session
    user = session.get('xxx') #获取session
    

    数据连接池

    当有多个函数需要用到sql语句时,需要多次操作数据库,就会多次的连接关闭数据库,会影响性能

    连接池实现原理:创建一个链接池,维护与数据库的连接,使用时来进行获取,使用完毕后在放回到连接池。

    安装dbutils和pymysql

    pip install dbutils
    pip install pymysql
    

    使用

    函数版的连接池

    import pymysql
    from DBUtils.PooledDB import PooledDB
    
    POOL = PooledDB(
        creator=pymysql,  # 使用链接数据库的模块
        maxconnections=6,  # 连接池允许的最大连接数,0和None表示不限制连接数
        mincached=2,  # 初始化时,链接池中至少创建的链接,0表示不创建
        blocking=True,  # 连接池中如果没有可用连接后,是否阻塞等待。True,等待;False,不等待然后报错
        ping=0,
        # ping MySQL服务端,检查是否服务可用。# 如:0 = None = never, 1 = default = whenever it is requested, 2 = when a cursor is created, 4 = when a query is executed, 7 = always
    
        host='127.0.0.1',
        port=3306,
        user='root',
        password='123',
        database='flask_db1',
        charset='utf8',
        autocommit=True   #提交sql语句
    )
    
    def sql(sql_info):
        conn = POOL.connection()
        sql1 = sql_info
        cursor1 = conn.cursor()
        cursor1.execute(sql1)
        r_all = cursor1.fetchall()
        cursor1.close()
        return r_all
    

    类版的连接池

    import pymysql
    from DBUtils.PooledDB import PooledDB
    
    
    class SqlHelp(object):
        def __int__(self,*args,**kwargs):
            self.pool = PooledDB(
                creator=pymysql,  # 使用链接数据库的模块
                maxconnections=6,  # 连接池允许的最大连接数,0和None表示不限制连接数
                mincached=2,  # 初始化时,链接池中至少创建的链接,0表示不创建
                blocking=True,  # 连接池中如果没有可用连接后,是否阻塞等待。True,等待;False,不等待然后报错
                ping=0,
                # ping MySQL服务端,检查是否服务可用。# 如:0 = None = never, 1 = default = whenever it is requested, 2 = when a cursor is created, 4 = when a query is executed, 7 = always
    
                host='127.0.0.1',
                port=3306,
                user='root',
                password='123',
                database='flask_db1',
                charset='utf8'
            )
    
        def open(self):
            conn = self.pool.connection()
            cursor1 = conn.cursor()
            return conn,cursor1
        
        def close(self,cursor1,conn):
            cursor1.close()
            conn.close()
    
        def sql(self,sql_info):
            conn,cursor1 = self.open()
            cursor1.execute(sql_info)
            r_all = cursor1.fetchall()
            self.close(cursor1,conn)
            return r_all
    
    db = SqlHelp()
    

    flask静态文件处理

    如果存放静态文件的文件名不是static,例如文件夹名为sssss,就需要写app = Flask(__name__,static_folder='sssss')
    
    from flask import Flask,render_template
    
    app = Flask(__name__,static_folder='static',static_url_path='/static')
    #static是存放配置文件的文件夹名,static_url_path是在路径的别名,不设置static_url_path之前在前端中这样写<img src="/static/index.png" alt="">,加了别名之后<img src="/img/index.png" alt="">
    
    建议前端img地址写法:<img src="{{ url_for('static',filename='index.png') }}" alt="">
    static下的index.png
    @app.route('/index')
    def index():
        return render_template('index.html')
    
    if __name__ == '__main__':
        app.run()
    

    WSGI源码分析

    1.程序启动,等待用户请求到来
    	app.run()  >>执行 run_simple(host, port, self, **options)# self就是我们创建的flask app
    2.用户请求到来会执行app.__call__方法,主要看app.__call__返回了什么  
    	2.1__call__返回了self.wsgi_app(environ, start_response)
        看wsgi_app返回了什么
        2.2 进到wsgi_app函数,其他的不看,就看该函数返回了什么,返回了response(environ, start_response),主要看response哪来的,自己就可以导入response,并返回
        2.3看response哪来的,上面有response..
        	response = self.full_dispatch_request()
            看full_dispatch_request返回什么response就是什么
            它返回了return self.finalize_request(rv)又调用了finalize_request(rv),rv就是视图的返回值,是个字符串
       2.4看finalize_request函数
    		response = self.make_response(rv)
       2.5在看make_response返回了什么
    	#跟据自己的内容进行判断
    	rv = self.response_class(rv, status=status, headers=headers)
        response_class = Response
        
        在这里进行一系列的判断并返回rv
    

    flask路由系统

    路由系统的两种写法

    # --第一种
    from flask import Flask,render_template
    
    
    app = Flask(__name__,static_folder='static')
    
    @app.route('/index')
    def index():
        return render_template('index.html')
    
    if __name__ == '__main__':
        app.run()
        
    # --第二种
    from flask import Flask,render_template
    
    
    app = Flask(__name__,static_folder='static')
    
    def index():
        return render_template('index.html')
    app.add_url_rule('/index','index',index)  #分别代表路径,别名,视图函数名
    
    if __name__ == '__main__':
        app.run()
    
    

    路由加载的源码流程

    - 将url和函数打包成为 rule 对象
    - 将rule对象添加到map对象中。 
    - app.url_map = map对象
    
    
    把路由和函数的对应关系放在flask的对象里,这个对象里保存了路由和函数的对应关系,如果有多个视图和路由, 将url和函数打包成为 rule 对象
    (放在Map对象里),可以通过app.url_map可以拿到map对象,对象里存放的
    就是url和函数名
    
    先执行route,返回了decorator,然后就执行decorator函数,内部执行了add_url_rule,帮助把url和函数的对应关系打包成为 rule 对象,放在Map对象中
    

    endpoint不写为什么默认是函数本身?

    源码:
    1.执行route,返回decorator
    2.
    def decorator(f): #f就是被装饰的函数
        endpoint = options.pop("endpoint", None)
        self.add_url_rule(rule, endpoint, f, **options)
    return f
    
    3.执行add_url_rule
    if endpoint is None:
    	endpoint = _endpoint_from_view_func(view_func)
        
    4.执行_endpoint_from_view_func(view_func)
    def _endpoint_from_view_func(view_func):
        assert view_func is not None, "expected view func if endpoint is not provided."
        return view_func.__name__ #返回的函数名
    

    动态路由

    @app.route('/login') 
    def login(): 
        return render_template('login.html') @app.route('/login/<name>') 
    def login(name): 
        print(type(name)) return render_template('login.html') @app.route('/login/<int:name>')  #数据类型的转换
    def login(name): 
        print(type(name)) 
        return render_template('login.html')
    

    flask的正则表达式的路由

    from flask import Flask,render_template 
    app = Flask(__name__) 
    from werkzeug.routing import BaseConverter 
    class RegConverter(BaseConverter): 
        def __init__(self, map, regex): 
            super().__init__(map) 
            self.regex = regex 
    app.url_map.converters['regex'] = RegConverter @app.route('/index/<regex("d+"):x1>') 
    def index(x1): 
        return render_template('index.html') 
    if __name__ == '__main__':
        app.run()
    

    flask视图

    FBV

    def index():
        return render_template('index.html') app.add_url_rule('/index', 'index', index) 
    
    @app.route('/login') 
    def login(): 
        return render_template('login.html')
    
    

    CBV

    from flask import Flask,render_template,views
    app = Flask(__name__,) 
    def test1(func): 
        def inner(*args,**kwargs):
            print('before1') 
            result = func(*args,**kwargs)
            print('after1')
            return result 
        return inner 
    def test2(func): 
    	def inner(*args,**kwargs): 
        	print('before2') 
        	result = func(*args,**kwargs) 
        	print('after2') 
        	return result 
    	return inner 
    class UserView(views.MethodView): 
        methods = ['GET',"POST"] 
        decorators = [test1,test2] 
        def get(self):
            print('get') 
            return 'get' 
        def post(self): 
            print('post') 
            return 'post' 
        app.add_url_rule('/user',view_func=UserView.as_view('user'))   #路由写法
    if __name__ == '__main__':
        app.run()
    
    

    模板

    基本用法

    from flask import Flask,render_template
    
    app = Flask(__name__)
    
    def fun(arg):
        return '你好'+arg
    
    @app.route('/index')
    def index():
        nums = [11,22,33,44]
        return render_template('index.html',nums=nums,f=fun) 
    #传递给前端函数的地址
    
    
    if __name__ == '__main__':
        app.run()
    
        
    
    

    index.html

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    <h1>index页面</h1>
    {{nums}}
    {{f("xx")}}   调用函数,并传递参数
    <img src="{{ url_for('static',filename='index.png') }}" alt="">
    </body>
    </html>
    
    

    继承

    from flask import Flask,render_template
    app = Flask(__name__)
    
    @app.route('/home')
    def home():
        return render_template('home.html')
    if __name__ == '__main__':
        app.run()
    
    

    layout.html模板

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
        <h1>模板</h1>
        {% block content %}
        {% endblock %}
    </body>
    </html>
    
    

    home.html

    {% extends 'layout.html' %}
    
    {% block content %}
    <input type="text">
    {% endblock %}
    
    

    定义全局模板的方法

    from flask import Flask,render_template
    app = Flask(__name__)
    
    @app.template_global() #定义全局
    def func(arg):
        return '等会' + arg
    
    @app.route('/index')
    def index():
        return render_template('index.html') #这就不用传给前端函数名
    
    if __name__ == '__main__':
        app.run()
    
    

    index.html

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Title</title>
    </head>
    <body>
    <h1>index页面</h1>
    {{func('你')}}  
    <img src="{{ url_for('static',filename='index.png') }}" alt="">
    </body>
    </html>
    
    

    flask中的特殊装饰器

    第一种情况

    from flask import Flask,render_template,request
    app = Flask(__name__)
    
    @app.before_request
    def f1():
        if request.path == '/login':
            return
        print('f1')
        
    
    @app.after_request
    def f10(response):
        print('f10')
        return response
    
    
    @app.route('/index')
    def index():
        print('index')
        return render_template('index.html')
    
    if __name__ == '__main__':
        app.run()
        
        
    结果:
    # f1
    # index
    # f10
    可以参考django的中间件的process_request和process_response的执行顺序
    
    

    第二种情况

    from flask import Flask,render_template,request
    app = Flask(__name__)
    
    @app.before_request
    def f1():
        if request.path == '/login':
            return
        print('f1')
        return '123'
    
    @app.after_request
    def f10(response):
        print('f10')
        return response
    
    @app.route('/index')
    def index():
        print('index')
        return render_template('index.html')
    
    if __name__ == '__main__':
        app.run()
    
    结果:
    # f1
    # f10
    
    

    多个装饰器

    from flask import Flask, render_template, request
    
    app = Flask(__name__)
    @app.before_request
    def f1():
        print('f1')
    
    @ app.before_request
    def f2():
        print('f2')
    
    @ app.after_request
    def f10(response):
        print('f10')
        return response
    
    @ app.after_request
    def f20(response):
        print('f20')
        return response
    
    @ app.route('/index')
    def index():
        print('index')
        return render_template('index.html')
    
    if __name__ == '__main__':
        app.run()
        app.__call__
        
    结果:
    f1
    f2
    index
    f20
    f10
    
    
  • 相关阅读:
    POJ2778 DNA Sequence AC自动机+矩阵二分
    POJ1204 Word Puzzles AC自动机 多串匹配
    与失散已久的小学同桌QQ聊天
    ZC公司员工评分系统——后台查询合成DataTable
    软考(2)编译原理
    GCT考试复习
    为自己加油!!!
    ZC公司员工评分系统——前台排版算法
    员工评分系统现场发布小感
    软考(3)操作系统
  • 原文地址:https://www.cnblogs.com/tangjian219/p/11912167.html
Copyright © 2011-2022 走看看