Flask的特点:
- pip install flask - 短小精悍、可扩展性强的 web框架 注意:上下文管理机制 - 依赖 wsgi:werkzeug
Flask的简单示例:
from flask import Flask # app = Flask(__name__,template_folder="templates",static_folder="static",static_url_path="/vvv") # template_folder(默认为 templates )、static_folder(默认为static)都是在这里设置;static_url_path="/vvv":设置静态文件的路径(给静态文件statci的路径设置一个别名;用在模板中) app = Flask(__name__) @app.route("/index") # 装饰器表示 路径和函数 绑定 def index(): return "hello world" if __name__ == "__main__": app.run() # 启动 socket
Flask登陆示例:
目录结构:
project
|-- static
|-- 1.png
|-- templates
|-- index.txt
|-- login.html
|-- flask_login.py
flask_login.py
from flask import Flask,request,render_template,redirect,session app = Flask(__name__) app.secret_key = "nbaoireunbadfo" # 设置session的时候,需要加盐加密 @app.route("/login",methods=["GET","POST"]) def login(): if request.method == "GET": return render_template("login.html") username = request.form.get("username") password = request.form.get("psw") if username == "neo" and password == "abc123": session["user"] = username # 设置 session return redirect("/index") else: return render_template("login.html", error="用户名密码错误") # 传递参数也可用字典 @app.route("/index") def index(): # 登陆之后才能访问 index 页面 if not session.get("user"): # 获取 session return redirect("/login") return render_template("index.txt") # 模板后缀不影响 if __name__ == "__main__": app.run() """ 小结: from flask import request # Flask 的请求数据是这样导入的,不是像Django 那样通过参数传递的 from flask import render_template # 渲染的模板 render_template(模板名,**{"传递的参数":传递的参数}) from flask import redirect # 重定向 from flask import session # session;flask中session默认是以加密的方式保存在了用户浏览器的cookie中;所以要设置加盐(加密) app.secret_key = "apiofdgqpweu" # session 加密加盐 request.methods # 获取请求方式 request.args # 获取 GET 请求的数据,对应 Django 的request.GET (url里面传过来的参数) request.form # 获取 POST 提交过来的数据,对应Django的 request.POST(请求体里面传过来的参数) @app.route("/login",methods=["GET","POST"]) # methods=["GET","POST"] 表示可接收GET和POST请求,默认只接收GET # 默认的 模板文件夹叫 templates;默认的静态文件叫 static # <form>表单只能发送 get 和 post请求,patch 和 delete请求要通过 Ajax发送 """
templates/index.txt
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <h3>欢迎登陆</h3> <img src="/static/1.png" alt=""> # src 中的 /static 可更改,通过 static_url_path 设置 </body> </html>
templates/login.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <form action="" method="post"> <input type="text" name="username"> <input type="password" name="psw"> <input type="submit" value="提交"> <span>{{error}}</span> <!-- {{ }} 用于传递参数 --> </form> </body> </html>
配置文件:
# 通过字符串找到类中的静态字段,如 "settings.Foo" 中的静态字段
# 通过字符串找到类中的静态字段,如 "settings.Foo" 中的静态字段 import importlib path = "settings.Foo" module_path,class_name = path.rsplit(".",1) module = importlib.import_module(module_path) cls = getattr(module,class_name) for key in dir(cls): # dir(cls) :返回一个列表,其中包含静态字段 if key.isupper(): # 静态字段得是大写 print(key,getattr(cls,key)) # 获取静态字段及对应的值
设置配置文件:
# settings.py class Config(object): 共用的配置信息 TESTING = False class ProductionConfig(Config): 生产环境下的配置 DEBUG = False class DevelopmentConfig(Config): DEBUG = True # app.py from flask import Flask app = Flask(__name__) # app.config # 获取 Flask 的 配置文件 # 配置文件中的字段必须得是 全大写 # app.config.from_object("python类或类的路径") # 设置配置文件 # 也可以这样单独设置配置信息: # app.config["DEBUG"] = True app.config.from_object("settings.DevelopmentConfig") if __name__ == "__main__": app.run()
其它配置:
flask中的配置文件是一个flask.config.Config对象(继承字典),默认配置为: { 'DEBUG': get_debug_flag(default=False), 是否开启Debug模式 'TESTING': False, 是否开启测试模式 'PROPAGATE_EXCEPTIONS': None, 'PRESERVE_CONTEXT_ON_EXCEPTION': None, 'SECRET_KEY': None, # session 相关(加盐加密) 'PERMANENT_SESSION_LIFETIME': timedelta(days=31), 'USE_X_SENDFILE': False, 'LOGGER_NAME': None, 'LOGGER_HANDLER_POLICY': 'always', 'SERVER_NAME': None, 'APPLICATION_ROOT': None, 'SESSION_COOKIE_NAME': 'session', 'SESSION_COOKIE_DOMAIN': None, 'SESSION_COOKIE_PATH': None, 'SESSION_COOKIE_HTTPONLY': True, 'SESSION_COOKIE_SECURE': False, 'SESSION_REFRESH_EACH_REQUEST': True, 'MAX_CONTENT_LENGTH': None, 'SEND_FILE_MAX_AGE_DEFAULT': timedelta(hours=12), 'TRAP_BAD_REQUEST_ERRORS': False, 'TRAP_HTTP_EXCEPTIONS': False, 'EXPLAIN_TEMPLATE_LOADING': False, 'PREFERRED_URL_SCHEME': 'http', 'JSON_AS_ASCII': True, 'JSON_SORT_KEYS': True, 'JSONIFY_PRETTYPRINT_REGULAR': True, 'JSONIFY_MIMETYPE': 'application/json', 'TEMPLATES_AUTO_RELOAD': None, }
路由系统
@app.route() 这个路由中的参数 endpoint="xx" :给该视图函数起一个别名(相当于django url中的name),如果 endpoint不写,默认是函数名;通过别名反解路径: url_for("xx") , url_for 需要导入 # 路由的动态传参 @app.route("/index/<int:nid>",method=["GET","POST"],endpoint="url1") # 路由中传参时,视图函数中需要接收一下;如果 int 不写,则默认是字符串格式; def index(nid): # 接收参数 print(url_for("url1",nid=777)) # 反解路径时,如果url中动态传参了,需要在后面指定传的参数 return "Index" # 动态传参时的类型包括:string,int,float 和 path; 如果不指定,默认是 string
请求和响应
from flask import Flask,request,render_template,redirect,make_response # ...(省略) @app.route("/login",methods=["GET","POST"]) def login(): """ 请求相关信息: request.method request.args request.form request.values request.cookies request.headers request.path # 请求的路径 request.full_path request.script_root request.url request.base_url request.url_root request.host_url request.host request.files # 文件 obj = request.files['the_file_name'] obj.save('/var/www/uploads/' + secure_filename(obj.filename)) # 保存文件 """ """ 响应相关信息: 1. 响应体: return "字符串" return jsonify(xxx) # 返回 json 数据; jsonify 需要从 flask导入 return render_template(模板) return redirect() # 重定向 上面4个返回的都是响应体(都是以响应体的方式返回) 2. 定制响应头(以 return "字符串" 为例): obj = make_response("Index") # make_response(你想要返回的内容); make_response() 能把你想要返回的内容进行封装到一个对象中; make_response 需要从 flask导入 obj.headers["xxx"] = "abc" # 设置响应头 return obj # 返回响应对象 3. 设置 cookie obj = make_response(render_template("index.html")) obj.set_cookie("key","value") # 设置cookie # cookie 是随着响应头给用户的 obj.delete_cookie("key") # 删除 cookie return obj """ # ...(省略)
模板渲染
python代码:
# 下面的字典模拟数据库 STU_DICT = { 1:{"name":"neo","age":22,"gender","male"}, 2:{"name":"maple","age":35,"gender","male"}, 3:{"name":"conan","age":12,"gender","male"}, } # index 函数 @app.route("/index") def index(): return render_template("index.html",stu_dict=STU_DICT) # 渲染模板时,传入要渲染的参数
对应的模板:
<table> <thead> <tr> <th>ID</th> <th>姓名</th> <th>年龄</th> <th>性别</th> </tr> </thead> <tbody> <!-- Flask 渲染模板时,用到的python函数需要加括号“()”;Django不需要 --> {% for k,v in stu_dict.items() %} <tr>{{k}}</tr> <!-- 可以用 .name 的方式 来取值;和Django的用法一样 --> <tr>{{v.name}}</tr> <!-- 可以用 v["age"] 的这种字典的方式 来取值 --> <tr>{{v["age"]}}</tr> <!-- 也可以用 v.get("gender","female") 的这种字典的方式 来取值; .get() 的时候也可设置默认值 --> <tr>{{v.get("gender")}}</tr> {% endfor %} </tbody> </table>
其它模板渲染的方法:
tpl_render.py
from flask import Flask,render_template,Markup app = Flask(__name__) def func(arg): return arg + 1 @app.template_global() # 如果某个函数在全局好多模板中都需要用到,为了避免多次传递这个函数,可以使用template_global()这个装饰器装饰该函数(该装饰器需要加() ),然后这个函数就能在全局所有模板中直接使用 def foo_global(a,b): # 使用方法:{{foo_global(2,3)}} return a + b @app.template_filter() # template_filter() 装饰的函数也能在全局(该装饰器需要加 ()),只是这个被装饰的函数 能用在前端的 if等语句中 def foo_filter(a,b,c): # 使用方法: return a + b + c @app.route("/tpl") def tpl(): context = { "user":["neo","maple"], "txt1":'<input type="text">', "txt2": '<input type="text">', "txt3": Markup('<input type="text">'), # HTML标签字符串也可以在后端 Markup(),从而让其在前端展示为 HTML标签 "func":func, # 也可以往前端传递一个函数;Flask传递函数时在前端不会自动执行,得自己调用(Django传递函数会自动执行) } return render_template("tpl.html",**context) # 可以通过 **context 的方式传参 if __name__ == "__main__": app.run()
tpl.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <!-- 和Django的用法一样,Flask也可以:list.索引;user是["neo","maple"]这个列表 --> {{user.0}} <!-- python语法也能在 Flask 中直接用:list[索引] --> {{user[0]}} <!-- {{txt1}}是字符串;Flask自动将 html 标签字符串做了处理 --> {{txt1}} <!-- 如果想让 HTML 标签字符串 变成HTML标签,可以在前端 |safe --> {{txt2|safe}} <!-- HTML标签字符串也可以在后端 Markup(),从而让其在前端展示为 HTML标签 --> {{txt3}} <!--传递函数--> {{func(2)}} <!-- template_global() 装饰的函数能在全局模板直接使用 --> {{foo_global(2,3)}} <!-- template_global() 装饰的函数能在全局模板直接使用,{{1|foo_filter(2,3)}} :a是1,b是2,c为3 --> {{1|foo_filter(2,3)}} <!--宏定义--> <!-- macro 是宏定义的语法(相当于函数定义的 def );input是宏定义的函数名(可随便改);type="text"表示type默认为"text",value="" 表示 value默认为空 --> {% macro input(name,type="text",value="") %} <h2>宏定义</h2> <input type="{{type}}" name="{{name}}" value="{{value}}"> {% endmacro %} <!--宏定义的函数不会自动执行,需要调用,如下--> {{input("neo")}} <!-- 此外, Flask 的 模板继承 和 include 的使用方法 和 Django 一样,不再重复 --> </body> </html>