Flask是一个基于Python开发并且依赖jinja2模板和Werkzeug WSGI服务的一个微型框架,对于Werkzeug本质是Socket服务端,其用于接收http请求并对请求进行预处理,然后触发Flask框架,开发人员基于Flask框架提供的功能对请求进行相应的处理,并返回给用户,如果要返回给用户复杂的内容时,需要借助jinja2模板来实现对模板的处理,即:将模板和数据进行渲染,将渲染后的字符串返回给用户浏览器。

“微”(micro) 并不表示你需要把整个 Web 应用塞进单个 Python 文件(虽然确实可以 ),也不意味着 Flask 在功能上有所欠缺。微框架中的“微”意味着 Flask 旨在保持核心简单而易于扩展。Flask 不会替你做出太多决策——比如使用何种数据库。而那些 Flask 所选择的——比如使用何种模板引擎——则很容易替换。除此之外的一切都由可由你掌握。如此,Flask 可以与您珠联璧合。

默认情况下,Flask 不包含数据库抽象层、表单验证,或是其它任何已有多种库可以胜任的功能。然而,Flask 支持用扩展来给应用添加这些功能,如同是 Flask 本身实现的一样。众多的扩展提供了数据库集成、表单验证、上传处理、各种各样的开放认证技术等功能。Flask 也许是“微小”的,但它已准备好在需求繁杂的生产环境中投入使用。

  1. pip3 install flask
  1. from werkzeug.wrappers import Request, Response
  2.  
  3. @Request.application
  4. def hello(request):
  5. return Response('Hello World!')
  6.  
  7. if __name__ == '__main__':
  8. from werkzeug.serving import run_simple
  9. run_simple('localhost', 4000, hello)

werkzeug

一. 基本使用

  1. from flask import Flask
  2. app = Flask(__name__)
  3.  
  4. @app.route('/')
  5. def hello_world():
  6. return 'Hello World!'
  7.  
  8. if __name__ == '__main__':
  9. app.run()

flask添加装饰器

  1. from flask import Flask
  2. import functools
  3.  
  4. def auth(func):
  5. @functools.wraps(func) # 保留函数的元信息
  6. def inner(*args,**kwargs):
  7. print('before')
  8. ret = func(*args,**kwargs)
  9. print('after')
  10. return ret
  11. return inner
  12.  
  13. app = Flask(__name__)
  14.  
  15. @app.route('/index')
  16. @auth
  17. def index():
  18. print('index')
  19. return 'index'
  20.  
  21. if __name__ == '__main__':
  22. app.run()

指定端口号

  1. from werkzeug.wrappers import Request, Response
  2. from werkzeug.serving import run_simple
  3.  
  4. @Request.application
  5. def hello(request):
  6. # ...
  7. # Flask框架
  8. #
  9. return Response('Hello World!')
  10.  
  11. if __name__ == '__main__':
  12.  
  13. run_simple('localhost', 4000, hello) # hello()

示例:before_request和after_request

  1. from flask import Flask,render_template,request,redirect,session
  2.  
  3. app = Flask(__name__)
  4.  
  5. app.secret_key = '39jrlasdfoajslfu8af'
  6.  
  7. # [b1,b2]
  8. @app.before_request
  9. def b1():
  10. print('b1')
  11. return 'gun'
  12.  
  13. @app.before_request
  14. def b2():
  15. print('b2')
  16.  
  17. # [a1,a2] -> [a2,a1]
  18. @app.after_request
  19. def a1(response):
  20. print('a1')
  21. return response
  22.  
  23. @app.after_request
  24. def a2(response):
  25. print('a2')
  26. return response
  27.  
  28. @app.route('/index')
  29. def index():
  30. print('index')
  31. return 'index'
  32.  
  33. if __name__ == '__main__':
  34. app.run()

基于装饰器做用户认证

  1. from flask import Flask,render_template,request,redirect,session
  2. app = Flask(__name__)
  3. app.secret_key = '39jrlasdfoajslfu8af'
  4. import functools
  5.  
  6. def auth(func):
  7. @functools.wraps(func) # 保留函数的元信息 一定要记得添加这行代码
  8. def inner(*args,**kwargs):
  9. if not session.get('user_info'):
  10. return redirect('/login')
  11. ret = func(*args,**kwargs)
  12. return ret
  13. return inner
  14.  
  15. @app.route('/login',methods=['GET','POST'])
  16. def login():
  17. if request.method == 'GET':
  18. return render_template('login.html')
  19.  
  20. user = request.form.get('user')
  21. pwd = request.form.get('pwd')
  22. if user == 'xiao' and pwd == 'xiao':
  23. session['user_info'] = user
  24. return redirect('/index')
  25. return render_template('login.html',msg="用户名或密码错误")
  26.  
  27. @app.route('/index')
  28. @auth
  29. def index():
  30. kuang_list = [
  31. {'id':1,'name':'name1','address':'昌平'},
  32. {'id':2,'name':'name2','address':'昌平'},
  33. {'id':3,'name':'name3','address':'昌平'},
  34. ]
  35.  
  36. return render_template('index.html',klist=kuang_list)
  37. # return render_template('index.html',**{'klist':kuang_list})
  38.  
  39. @app.route('/detail')
  40. @auth
  41. def detail():
  42. nid = request.args.get('nid')
  43. return render_template('detail.html')
  44.  
  45. @app.route('/logout')
  46. def logout():
  47. del session['user_info']
  48. return redirect('/login')
  49.  
  50. if __name__ == '__main__':
  51. app.run()

基于before_request实现用户认证

  1. from flask import Flask,render_template,request,redirect,session
  2. app = Flask(__name__)
  3. app.secret_key = '39jrlasdfoajslfu8af'
  4.  
  5. @app.before_request
  6. def xxxxxxxxxxxxxx():
  7. if request.path == '/login':
  8. return None
  9.  
  10. if not session.get('user_info'):
  11. return redirect('/login')
  12.  
  13. @app.route('/login',methods=['GET','POST'])
  14. def login():
  15. print('我是login')
  16. if request.method == 'GET':
  17. return render_template('login.html')
  18. user = request.form.get('user')
  19. pwd = request.form.get('pwd')
  20. if user == 'xiao' and pwd == 'xiao':
  21. session['user_info'] = user
  22. return redirect('/index')
  23. return render_template('login.html',msg="用户名或密码错误")
  24.  
  25. @app.route('/index')
  26. def index():
  27. print('我是index')
  28. kuang_list = [
  29. {'id':1,'name':'name1','address':'昌平'},
  30. {'id':2,'name':'name2','address':'昌平'},
  31. {'id':3,'name':'name3','address':'昌平'},
  32. ]
  33.  
  34. return render_template('index.html',klist=kuang_list) # 这种传参方式,要注意
  35. # return render_template('index.html',**{'klist':kuang_list}) # 这种传参方式,要加**打散
  36.  
  37. @app.route('/detail')
  38. def detail():
  39. nid = request.args.get('nid')
  40. return render_template('detail.html')
  41.  
  42. @app.route('/logout')
  43. def logout():
  44. del session['user_info']
  45.  
  46. return redirect('/login')
  47.  
  48. if __name__ == '__main__':
  49. app.run()

二、配置文件

  1. flask中的配置文件是一个flask.config.Config对象(继承字典),默认配置为:
  2. {
  3. 'DEBUG': get_debug_flag(default=False), 是否开启Debug模式
  4. 'TESTING': False, 是否开启测试模式
  5. 'PROPAGATE_EXCEPTIONS': None,
  6. 'PRESERVE_CONTEXT_ON_EXCEPTION': None,
  7. 'SECRET_KEY': None,
  8. 'PERMANENT_SESSION_LIFETIME': timedelta(days=31),
  9. 'USE_X_SENDFILE': False,
  10. 'LOGGER_NAME': None,
  11. 'LOGGER_HANDLER_POLICY': 'always',
  12. 'SERVER_NAME': None,
  13. 'APPLICATION_ROOT': None,
  14. 'SESSION_COOKIE_NAME': 'session',
  15. 'SESSION_COOKIE_DOMAIN': None,
  16. 'SESSION_COOKIE_PATH': None,
  17. 'SESSION_COOKIE_HTTPONLY': True,
  18. 'SESSION_COOKIE_SECURE': False,
  19. 'SESSION_REFRESH_EACH_REQUEST': True,
  20. 'MAX_CONTENT_LENGTH': None,
  21. 'SEND_FILE_MAX_AGE_DEFAULT': timedelta(hours=12),
  22. 'TRAP_BAD_REQUEST_ERRORS': False,
  23. 'TRAP_HTTP_EXCEPTIONS': False,
  24. 'EXPLAIN_TEMPLATE_LOADING': False,
  25. 'PREFERRED_URL_SCHEME': 'http',
  26. 'JSON_AS_ASCII': True,
  27. 'JSON_SORT_KEYS': True,
  28. 'JSONIFY_PRETTYPRINT_REGULAR': True,
  29. 'JSONIFY_MIMETYPE': 'application/json',
  30. 'TEMPLATES_AUTO_RELOAD': None,
  31. }
  32.  
  33. 方式一:
  34. app.config['DEBUG'] = True
  35.  
  36. PS 由于Config对象本质上是字典,所以还可以使用app.config.update(...)
  37.  
  38. 方式二:
  39. app.config.from_pyfile("python文件名称")
  40. 如:
  41. settings.py
  42. DEBUG = True
  43.  
  44. app.config.from_pyfile("settings.py")
  45.  
  46. app.config.from_envvar("环境变量名称")
  47. 环境变量的值为python文件名称名称,内部调用from_pyfile方法
  48.  
  49. app.config.from_json("json文件名称")
  50. JSON文件名称,必须是json格式,因为内部会执行json.loads
  51.  
  52. app.config.from_mapping({'DEBUG':True})
  53. 字典格式
  54.  
  55. app.config.from_object("python类或类的路径") # 写在flask启动的py文件里面进行配置版本
  56.  
  57. app.config.from_object('pro_flask.settings.TestingConfig')
  58.  
  59. settings.py
  60.  
  61. class Config(object): # 基础版本
  62. DEBUG = False
  63. TESTING = False
  64. DATABASE_URI = 'sqlite://:memory:'
  65.  
  66. class ProductionConfig(Config): # 生产环境,线下环境
  67. DATABASE_URI = 'mysql://user@localhost/foo'
  68.  
  69. class DevelopmentConfig(Config): # 开发环境
  70. DEBUG = True
  71.  
  72. class TestingConfig(Config): # 测试环境 各个环境下连接的数据库是不一样的
  73. TESTING = True
  74.  
  75. PS: sys.path中已经存在路径开始写
  76.  
  77. PS: settings.py文件默认路径要放在程序root_path目录,如果instance_relative_configTrue,则就是instance_path目录

三、路由系统

  • @app.route('/user/<username>')  # 传字符串   他们的内部都对应一个转换器
  • @app.route('/post/<int:post_id>')   # 传数字
  • @app.route('/post/<float:post_id>')  # 浮点数
  • @app.route('/post/<path:path>')   # 路径,拼url
  • @app.route('/login', methods=['GET', 'POST']),method列表里面添加的允许的请求方式,如果不添加method列表,则只允许GET请求。

常用路由系统有以上五种,所有的路由系统都是基于一下对应关系来处理:

  1. DEFAULT_CONVERTERS = {
  2. 'default': UnicodeConverter,
  3. 'string': UnicodeConverter,
  4. 'any': AnyConverter,
  5. 'path': PathConverter,
  6. 'int': IntegerConverter,
  7. 'float': FloatConverter,
  8. 'uuid': UUIDConverter,
  9. }
  1. def auth(func):
  2. def inner(*args, **kwargs):
  3. print('before')
  4. result = func(*args, **kwargs)
  5. print('after')
  6. return result
  7.  
  8. return inner
  9.  
  10. @app.route('/index.html',methods=['GET','POST'],endpoint='index')
  11. @auth
  12. def index():
  13. return 'Index'
  14.  
  15.  
  16. def index():
  17. return "Index"
  18.  
  19. self.add_url_rule(rule='/index.html', endpoint="index", view_func=index, methods=["GET","POST"])
  20. or
  21. app.add_url_rule(rule='/index.html', endpoint="index", view_func=index, methods=["GET","POST"])
  22. app.view_functions['index'] = index
  23.  

  24. def auth(func):
  25. def inner(*args, **kwargs):
  26. print('before')
  27. result = func(*args, **kwargs)
  28. print('after')
  29. return result
  30.  
  31. return inner
  32.  
  33. class IndexView(views.View):
  34. methods = ['GET']
  35. decorators = [auth, ]
  36.  
  37. def dispatch_request(self):
  38. print('Index')
  39. return 'Index!'
  40.  
  41. app.add_url_rule('/index', view_func=IndexView.as_view(name='index')) # name=endpoint
  42.  
  43.  
  44. class IndexView(views.MethodView):
  45. methods = ['GET']
  46. decorators = [auth, ]
  47.  
  48. def get(self):
  49. return 'Index.GET'
  50.  
  51. def post(self):
  52. return 'Index.POST'
  53.  
  54. app.add_url_rule('/index', view_func=IndexView.as_view(name='index')) # name=endpoint
  55.  
  56. @app.routeapp.add_url_rule参数:
  57. rule, URL规则
  58. view_func, 视图函数名称
  59. defaults=None, 默认值,当URL中无参数,函数需要参数时,使用defaults={'k':'v'}为函数提供参数
  60. endpoint=None, 名称,用于反向生成URL,即: url_for('名称')
  61. methods=None, 允许的请求方式,如:["GET","POST"]
  62.  
  63. strict_slashes=None, URL最后的 / 符号是否严格要求,
  64. 如:
  65. @app.route('/index',strict_slashes=False),
  66. 访问 http://www.xx.com/index/ 或 http://www.xx.com/index均可
  67. @app.route('/index',strict_slashes=True)
  68. 仅访问 http://www.xx.com/index
  69. redirect_to=None, 重定向到指定地址
  70. 如:
  71. @app.route('/index/<int:nid>', redirect_to='/home/<nid>')

  72. def func(adapter, nid):
  73. return "/home/888"
  74. @app.route('/index/<int:nid>', redirect_to=func)
  75. subdomain=None, 子域名访问
  76. from flask import Flask, views, url_for
  77.  
  78. app = Flask(import_name=__name__)
  79. app.config['SERVER_NAME'] = 'wupeiqi.com:5000'
  80.  
  81. @app.route("/", subdomain="admin")
  82. def static_index():
  83. """Flask supports static subdomains
  84. This is available at static.your-domain.tld"""
  85. return "static.your-domain.tld"
  86.  
  87. @app.route("/dynamic", subdomain="<username>")
  88. def username_index(username):
  89. """Dynamic subdomains are also supported
  90. Try going to user1.your-domain.tld/dynamic"""
  91. return username + ".your-domain.tld"
  92.  
  93. if __name__ == '__main__':
  94. app.run()

a.注册路由原理

  1. from flask import Flask, views, url_for
  2. from werkzeug.routing import BaseConverter
  3.  
  4. app = Flask(import_name=__name__)
  5.  
  6. class RegexConverter(BaseConverter):
  7. """
  8. 自定义URL匹配正则表达式
  9. """
  10. def __init__(self, map, regex):
  11. super(RegexConverter, self).__init__(map)
  12. self.regex = regex
  13.  
  14. def to_python(self, value):
  15. """
  16. 路由匹配时,匹配成功后传递给视图函数中参数的值
  17. :param value:
  18. :return:
  19. """
  20. return int(value)
  21.  
  22. def to_url(self, value):
  23. """
  24. 使用url_for反向生成URL时,传递的参数经过该方法处理,返回的值用于生成URL中的参数
  25. :param value:
  26. :return:
  27. """
  28. val = super(RegexConverter, self).to_url(value)
  29. return val
  30.  
  31. # 添加到flask中
  32. app.url_map.converters['regex'] = RegexConverter
  33.  
  34. @app.route('/index/<regex("\d+"):nid>')
  35. def index(nid):
  36. print(url_for('index', nid=''))
  37. return 'Index'
  38.  
  39. if __name__ == '__main__':
  40. app.run()

b. 自定制正则路由匹配

  1. app.url_map.converters['regex'] = RegexConverter
  2. # 这个写在启动py文件里,告诉flask我自己要定义一个路由转换器啦,名字叫regex
  3. # 转换器对应的类就是RegexConverter
  4.  
  5. class RegexConverter(BaseConverter):
  6. """
  7. 自定义URL匹配正则表达式
  8. """
  9. def __init__(self, map, regex):# 参数regex接收传过来的nid参数
  10. super(RegexConverter, self).__init__(map)
  11. self.regex = regex
  12.  
  13. def to_python(self, value):
  14. """
  15. 路由匹配时,匹配成功后传递给视图函数中参数的值
  16. :param value:
  17. :return:
  18. """
  19. return int(value)
  20.  
  21. def to_url(self, value):
  22. """
  23. 使用url_for反向生成URL时,传递的参数经过该方法处理,返回的值用于生成URL中的参数
  24. :param value:
  25. :return:
  26. """
  27. val = super(RegexConverter, self).to_url(value)
  28. return val
  29.  
  30. app.url_map.converters['xxx'] = RegexConverter
  31.  
  32. @app.route('/index/<xxx("\d+"):nid>',methods=['GET','POST']) # 定义好转换器之后,就这样写了。
  33. def index(nid):
  34. print(nid,type(nid))
  35. v = url_for('index',nid=999) # /index/999
  36. print(v)
  37. return "Index"

添加路由的两种方式

  1. # 路由方式一(*):
  2. @app.route('/index',methods=['GET','POST'])
  3. def index():
  4. return "Index"
  5.  
  6. # 路由方式二:
  7. def order():
  8. return 'Order'
  9.  
  10. app.add_url_rule('/order',view_func=order)

反向生成url

  1. @app.route('/index',methods=['GET','POST'],endpoint='n1') # endpoint 给路由起别名。如果不起别名,就用函数名
  2. def index():
  3. v1 = url_for('n1')
  4. v2 = url_for('login') # url_for 反向生成url
  5. v3 = url_for('logout')
  6. print(v1,v2,v3)
  7. return "Index"
  8.  
  9. @app.route('/login',methods=['GET','POST'])
  10. def login():
  11. return "login"
  12.  
  13. @app.route('/logout',methods=['GET','POST'])
  14. def logout():
  15. return "logout"

四、模板

1、模板的使用

Flask使用的是Jinja2模板,所以其语法和Django无差别

2、自定义模板方法

Flask中自定义模板方法的方式和Bottle相似,创建一个函数并通过参数的形式传入render_template,如:

  1. <!DOCTYPE html>
  2. <html>
  3. <head lang="en">
  4. <meta charset="UTF-8">
  5. <title></title>
  6. </head>
  7. <body>
  8. <h1>自定义函数</h1>
  9. {{ww()|safe}}
  10.  
  11. </body>
  12. </html>

html

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. from flask import Flask,render_template
  4. app = Flask(__name__)
  5.  
  6. def wupeiqi():
  7. return '<h1>Wupeiqi</h1>'
  8.  
  9. @app.route('/login', methods=['GET', 'POST'])
  10. def login():
  11. return render_template('login.html', ww=wupeiqi)
  12.  
  13. app.run()

run.py

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Title</title>
  6. </head>
  7. <body>
  8.  
  9. {% macro input(name, type='text', value='') %}
  10. <input type="{{ type }}" name="{{ name }}" value="{{ value }}">
  11. {% endmacro %}
  12.  
  13. {{ input('n1') }}
  14.  
  15. {% include 'tp.html' %}
  16.  
  17. <h1>asdf{{ v.k1}}</h1>
  18. </body>
  19. </html>

其他

注意:Markup等价django的mark_safe

五、请求和响应

  1. from flask import Flask
  2. from flask import request
  3. from flask import render_template
  4. from flask import redirect
  5. from flask import make_response
  6.  
  7. app = Flask(__name__)
  8.  
  9. @app.route('/login.html', methods=['GET', "POST"])
  10. def login():
  11.  
  12. # 请求相关信息
  13. # request.method 请求的方法
  14. # request.args 获取GET请求的内容
  15. # request.args.get("id") 获取id的值
  16. # request.args。getlist('ids') 获取多个值
  17.  
  18. # data = request.args
  19. # data_dict = data.to_dict() 通过to_dict()方法转化成字典
  20.  
  21. # request.remote_addr 获取ip
  22.  
  23. # request.form 获取POST请求的值,获取form表单里面的内容
  24. # request.values 所有的请求参数
  25. # request.cookies 获取cookies
  26. # request.headers 获取headers信息,是个list
  27. # request.path 仅获取端口号和参数中间的路由 /testrequest
  28. # request.full_path
  29. # request.script_root 什么都没获取到
  30. # request.url 获取完整的url,带参数 http://192.168.1.183:5000/testrequest?a&b
  31. # request.base_url 获取除参数外的url部分 http://192.168.1.183:5000/testrequest
  32. # request.url_root 仅获取地址和端口号 http://192.168.1.183:5000/
  33. # request.host_url
  34. # request.host
  35. # request.files 随请求上传的文件
  36. # obj = request.files['the_file_name']
  37. # obj.save('/var/www/uploads/' + secure_filename(f.filename))
  38.  
  39. # request.data 是请求的数据
  40.  
  41. # 响应相关信息
  42. # return "字符串" 实际上是调用make_response
  43. # return render_template('html模板路径',**{}) **{'obj':obj}形式
  44. # return render_template('html模板路径',obj=obj)
  45. # return redirect('/index.html')
  46. # return jsonify(**{'obj':obj}) 返回json数据 from flask import jsonify
  47.  
  48. # response = make_response(render_template('index.html'))
  49. # response是flask.wrappers.Response类型
  50. # response.delete_cookie('key')
  51. # response.set_cookie('key', 'value')
  52. # response.headers['X-Something'] = 'A value'
  53. # return response
  54.  
  55. return "内容"
  56.  
  57. if __name__ == '__main__':
  58. app.run()

六、Session

除请求对象之外,还有一个 session 对象。它允许你在不同请求间存储特定用户的信息。它是在 Cookies 的基础上实现的,并且对 Cookies 进行密钥签名要使用会话,你需要设置一个密钥。

  • 设置:session['username'] = 'xxx'

  • 删除:session.pop('username', None)
  1. from flask import Flask, session, redirect, url_for, escape, request
  2.  
  3. app = Flask(__name__)
  4.  
  5. @app.route('/')
  6. def index():
  7. if 'username' in session:
  8. return 'Logged in as %s' % escape(session['username'])
  9. return 'You are not logged in'
  10.  
  11. @app.route('/login', methods=['GET', 'POST'])
  12. def login():
  13. if request.method == 'POST':
  14. session['username'] = request.form['username']
  15. return redirect(url_for('index'))
  16. return '''
  17. <form action="" method="post">
  18. <p><input type=text name=username>
  19. <p><input type=submit value=Login>
  20. </form>
  21. '''
  22.  
  23. @app.route('/logout')
  24. def logout():
  25. # remove the username from the session if it's there
  26. session.pop('username', None)
  27. return redirect(url_for('index'))
  28.  
  29. # set the secret key. keep this really secret:
  30. app.secret_key = 'A0Zr98j/3yX R~XHH!jmN]LWX/,?RT'

基本使用

  1. pip3 install Flask-Session
  2.  
  3. run.py
  4. from flask import Flask
  5. from flask import session
  6. from pro_flask.utils.session import MySessionInterface
  7. app = Flask(__name__)
  8.  
  9. app.secret_key = 'A0Zr98j/3yX R~XHH!jmN]LWX/,?RT'
  10. app.session_interface = MySessionInterface()
  11.  
  12. @app.route('/login.html', methods=['GET', "POST"])
  13. def login():
  14. print(session)
  15. session['user1'] = 'alex'
  16. session['user2'] = 'alex'
  17. del session['user2']
  18.  
  19. return "内容"
  20.  
  21. if __name__ == '__main__':
  22. app.run()
  23.  
  24. session.py
  25. #!/usr/bin/env python
  26. # -*- coding:utf-8 -*-
  27. import uuid
  28. import json
  29. from flask.sessions import SessionInterface
  30. from flask.sessions import SessionMixin
  31. from itsdangerous import Signer, BadSignature, want_bytes
  32.  
  33. class MySession(dict, SessionMixin):
  34. def __init__(self, initial=None, sid=None):
  35. self.sid = sid
  36. self.initial = initial
  37. super(MySession, self).__init__(initial or ())
  38.  
  39. def __setitem__(self, key, value):
  40. super(MySession, self).__setitem__(key, value)
  41.  
  42. def __getitem__(self, item):
  43. return super(MySession, self).__getitem__(item)
  44.  
  45. def __delitem__(self, key):
  46. super(MySession, self).__delitem__(key)
  47.  
  48. class MySessionInterface(SessionInterface):
  49. session_class = MySession
  50. container = {}
  51.  
  52. def __init__(self):
  53. import redis
  54. self.redis = redis.Redis()
  55.  
  56. def _generate_sid(self):
  57. return str(uuid.uuid4())
  58.  
  59. def _get_signer(self, app):
  60. if not app.secret_key:
  61. return None
  62. return Signer(app.secret_key, salt='flask-session',
  63. key_derivation='hmac')
  64.  
  65. def open_session(self, app, request):
  66. """
  67. 程序刚启动时执行,需要返回一个session对象
  68. """
  69. sid = request.cookies.get(app.session_cookie_name)
  70. if not sid:
  71. sid = self._generate_sid()
  72. return self.session_class(sid=sid)
  73.  
  74. signer = self._get_signer(app)
  75. try:
  76. sid_as_bytes = signer.unsign(sid)
  77. sid = sid_as_bytes.decode()
  78. except BadSignature:
  79. sid = self._generate_sid()
  80. return self.session_class(sid=sid)
  81.  
  82. # session保存在redis中
  83. # val = self.redis.get(sid)
  84. # session保存在内存中
  85. val = self.container.get(sid)
  86.  
  87. if val is not None:
  88. try:
  89. data = json.loads(val)
  90. return self.session_class(data, sid=sid)
  91. except:
  92. return self.session_class(sid=sid)
  93. return self.session_class(sid=sid)
  94.  
  95. def save_session(self, app, session, response):
  96. """
  97. 程序结束前执行,可以保存session中所有的值
  98. 如:
  99. 保存到resit
  100. 写入到用户cookie
  101. """
  102. domain = self.get_cookie_domain(app)
  103. path = self.get_cookie_path(app)
  104. httponly = self.get_cookie_httponly(app)
  105. secure = self.get_cookie_secure(app)
  106. expires = self.get_expiration_time(app, session)
  107.  
  108. val = json.dumps(dict(session))
  109.  
  110. # session保存在redis中
  111. # self.redis.setex(name=session.sid, value=val, time=app.permanent_session_lifetime)
  112. # session保存在内存中
  113. self.container.setdefault(session.sid, val)
  114.  
  115. session_id = self._get_signer(app).sign(want_bytes(session.sid))
  116.  
  117. response.set_cookie(app.session_cookie_name, session_id,
  118. expires=expires, httponly=httponly,
  119. domain=domain, path=path, secure=secure)

自定义Session

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. """
  4. pip3 install redis
  5. pip3 install flask-session
  6.  
  7. """
  8.  
  9. from flask import Flask, session, redirect
  10. from flask.ext.session import Session
  11.  
  12. app = Flask(__name__)
  13. app.debug = True
  14. app.secret_key = 'asdfasdfasd'
  15.  
  16. app.config['SESSION_TYPE'] = 'redis'
  17. from redis import Redis
  18. app.config['SESSION_REDIS'] = Redis(host='192.168.0.94',port='')
  19. Session(app)
  20.  
  21. @app.route('/login')
  22. def login():
  23. session['username'] = 'alex'
  24. return redirect('/index')
  25.  
  26. @app.route('/index')
  27. def index():
  28. name = session['username']
  29. return name
  30.  
  31. if __name__ == '__main__':
  32. app.run()

第三方session

七、蓝图

蓝图用于为应用提供目录划分:

小型应用程序:示例

大型应用程序:示例

其他:

  • 蓝图URL前缀:xxx = Blueprint('account', __name__,url_prefix='/xxx')
  • 蓝图子域名:xxx = Blueprint('account', __name__,subdomain='admin')
    # 前提需要给配置SERVER_NAME: app.config['SERVER_NAME'] = 'wupeiqi.com:5000'
    # 访问时:admin.wupeiqi.com:5000/login.html

八、message

message是一个基于Session实现的用于保存数据的集合,其特点是:使用一次就删除。闪现

  1. from flask import Flask, flash, redirect, render_template, request, get_flashed_messages
  2.  
  3. app = Flask(__name__)
  4. app.secret_key = 'some_secret'
  5.  
  6. @app.route('/')
  7. def index1():
  8. messages = get_flashed_messages()
  9. print(messages)
  10. return "Index1"
  11.  
  12. @app.route('/set')
  13. def index2():
  14. v = request.args.get('p')
  15. flash(v)
  16. return 'ok'
  17.  
  18. if __name__ == "__main__":
  19. app.run()

九、中间件

  1. from flask import Flask, flash, redirect, render_template, request
  2.  
  3. app = Flask(__name__)
  4. app.secret_key = 'some_secret'
  5.  
  6. @app.route('/')
  7. def index1():
  8. return render_template('index.html')
  9.  
  10. @app.route('/set')
  11. def index2():
  12. v = request.args.get('p')
  13. flash(v)
  14. return 'ok'
  15.  
  16. class MiddleWare:
  17. def __init__(self,wsgi_app):
  18. self.wsgi_app = wsgi_app
  19.  
  20. def __call__(self, *args, **kwargs):
  21.  
  22. return self.wsgi_app(*args, **kwargs)
  23.  
  24. if __name__ == "__main__":
  25. app.wsgi_app = MiddleWare(app.wsgi_app)
  26. app.run(port=9999)

十、请求扩展

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. from flask import Flask, Request, render_template
  4.  
  5. app = Flask(__name__, template_folder='templates')
  6. app.debug = True
  7.  
  8. @app.before_first_request
  9. def before_first_request1():
  10. print('before_first_request1')
  11.  
  12. @app.before_first_request
  13. def before_first_request2():
  14. print('before_first_request2')
  15.  
  16. @app.before_request
  17. def before_request1():
  18. Request.nnn = 123
  19. print('before_request1')
  20.  
  21. @app.before_request
  22. def before_request2():
  23. print('before_request2')
  24.  
  25. @app.after_request
  26. def after_request1(response):
  27. print('before_request1', response)
  28. return response
  29.  
  30. @app.after_request
  31. def after_request2(response):
  32. print('before_request2', response)
  33. return response
  34.  
  35. @app.errorhandler(404)
  36. def page_not_found(error):
  37. return 'This page does not exist', 404
  38.  
  39. @app.template_global()
  40. def sb(a1, a2):
  41. return a1 + a2
  42.  
  43. @app.template_filter()
  44. def db(a1, a2, a3):
  45. return a1 + a2 + a3
  46.  
  47. @app.route('/')
  48. def hello_world():
  49. return render_template('hello.html')
  50.  
  51. if __name__ == '__main__':
  52. app.run()
  1. 调用方式:{{sb(1,2)}} {{ 1|db(2,3)}}

十一、Flask插件

  • WTForms
  • SQLAchemy
  • 等...    http://flask.pocoo.org/extensions/

补充:

  1. pip install pipreqs
    安装这个模块之后项目里会自动生成一个requirements.txt文件,里面显示的是这个项目安装的模块及其版本信息。
    生成依赖文件requirements.txt 命令: pipreqs ./ (./表示当前文件)
    安装依赖文件requirements.txt 命令: pip install -r requirements.txt

Flask详解的更多相关文章

  1. flask基础之session原理详解(十)

    前言 flask_session是flask框架实现session功能的一个插件,用来替代flask自带的session实现机制,flask默认的session信息保存在cookie中,不够安全和灵活 ...

  2. Flask request 属性详解

    Flask request 属性详解 一.关于request在Flask的官方文档中是这样介绍request的:对于 Web 应用,与客户端发送给服务器的数据交互至关重要.在 Flask 中由全局的 ...

  3. Flask(4)- URL 组成部分详解

    URL Uniform Resource Locator 的简写,中文名叫统一资源定位符 用于表示服务端的各种资源,例如网页 下面将讲解 Flask 中如何提取组成 URL 的各个部分   URL 组 ...

  4. Docker系列10—容器编排工具Docker Compose详解

    本文收录在容器技术学习系列文章总目录 1.Docker Compose 概述 Compose是一个用于定义和运行多容器Docker应用程序的工具.使用Compose,您可以使用Compose文件来配置 ...

  5. Bootstrap如何实现导航条?导航条实例详解

    本文主要和大家分享Bootstrap实现导航实例详解,在建设一个网站的时候,不同的页面有很多元素是一样的,比如导航条.侧边栏等,我们可以使用模板的继承,避免重复编写html代码.现在我们打算实现一个在 ...

  6. nginx入门与实战 安装 启动 配置nginx Nginx状态信息(status)配置 正向代理 反向代理 nginx语法之location详解

    nginx入门与实战 网站服务 想必我们大多数人都是通过访问网站而开始接触互联网的吧.我们平时访问的网站服务 就是 Web 网络服务,一般是指允许用户通过浏览器访问到互联网中各种资源的服务. Web ...

  7. Linux pip安装使用详解

    简介 pip是Python有它自己的包管理工具,与yum和apt-get相似. 安装步骤: 1.下载get-pip.py:https://bootstrap.pypa.io/get-pip.py 2. ...

  8. (转)Nginx静态服务配置---详解root和alias指令

    Nginx静态服务配置---详解root和alias指令 原文:https://www.jianshu.com/p/4be0d5882ec5 静态文件 Nginx以其高性能著称,常用与做前端反向代理服 ...

  9. celery详解

    目录 Celery详解 1.背景 2.形象比喻 3.celery具体介绍 3.1 Broker 3.2 Backend 4.使用 4.1 celery架构 4.2 安装redis+celery 4.3 ...

随机推荐

  1. Mapper method 'com.autoyol.mapper.trans.AccountLogMapper.getTotalIncomByMemNoLastest attempted to return null from a method with a primitive return type (int).解决方法

    1.打开日志输出,降低日志级别. <AppenderRef ref="console" level="trace"></AppenderRef ...

  2. AES五种加密模式

    分组密码在加密时明文分组的长度是固定的,而实用中待加密消息的数据量是不定的,数据格式可能是多种多样的.为了能在各种应用场合安全地使用分组密码,通常对不同的使用目的运用不同的工作模式. 一.电码本模式( ...

  3. VSCode------.net core2.0发布后配置到Window Service 2008R2报500错误

    如图: 解决方法: 出现这个错误是因为 IIS 采用了更安全的 web.config 管理机制,默认情况下会锁住配置项不允许更改. 要取消锁定可以运行命令行 %windir%\system32\ine ...

  4. Cookie 和 Session机制具体解释

    原文地址:http://blog.csdn.net/fangaoxin/article/details/6952954     会话(Session)跟踪是Web程序中经常使用的技术,用来跟踪用户的整 ...

  5. 解压安装的tomcat, 使用chkconfig命令让tomcat 随机启动,tomcat 变为系统服务

    使用解压安装的tomcat包,命令行输入 service tomcat start 会报 tomcat: unrecognized service 错误提示,意思是说系统没有找到该服务. 好了,我们现 ...

  6. vue实现百度搜索下拉提示功能

    这段代码用到vuejs和vue-resouece.实现对接智能提示接口,并通过上下键选择提示项,按enter进行搜索 <!DOCTYPE html> <html lang=" ...

  7. git push 问题汇总

    Q:git push时卡死 这个问题找的快要放弃的时候... A: git config --global http.postBuffer [via] Q:git push 报错 Counting o ...

  8. java.lang.IllegalArgumentException: No converter found for return value of type

    原文地址: http://blog.csdn.net/linhaiguo/article/details/51554766 问题原因: 请求返回的数据无法转换,需要添加如下配置 解决方法: 1.在po ...

  9. React Native(十一)——按钮重复点击事件的处理

    最初开始做后台管理系统的时候,就曾遇到过这样一种场景:"快速点击确认按钮,就会对此触发确认事件,导致多次请求数据库":于是最终我们得当的通过处理那个确认button,解决了上述问题 ...

  10. js 简易模板引擎 , 持续更新。。。

    <script> var _mytpl = (function(){ var _verson = 1.0; return { _data:{}, load:function(html,da ...