目录
一.类视图
二.蓝图
在Flask中,可通过视图函数展示视图
http://t.csdnimg.cn/r6IFG
也可基于类实现,类视图的好处是支持继承。标准类视图是继承flask.views模块中基类View的子类
from flask.views import View
该子类中必须重写View类中的dispatch_request()方法。除dispatch_request()方法外,我们也可以根据需要向View子类中添加其他方法或属性。
•methods属性:设置当前类视图可以处理的请求方式列表。
•dispatch_request()方法:用于实现处理不同HTTP请求的具体逻辑,该方法可以通过关键字参数接收URL传递的参数。
•as_view()方法:用于将类转换为实际视图函数。as_view()方法必须传入一个name参数,用于指定生成的视图函数名称,也可以根据需要传入一些关键字参数,这些参数都会转发给类的构造方法,以创建类的实例,并调用类内部的dispatch_request()方法。
如果希望类视图能够对浏览器发送的HTTP请求进行处理,那么需要将类视图与URL建立映射关系。我们需要通过add_url_rule()方法将类视图与URL进行映射,不过该方法的view_func参数不能直接传入类视图的名称,而是需要传入通过as_view()方法将类视图转换后的视图函数。
from flask import Flask
from flask.views import View
class MyView(View): # 定义类视图
def dispatch_request(self, name):# 重写dispatch_request()方法
return f'hello {name}'
app = Flask(__name__)
# 将类视图与URL规则进行映射
app.add_url_rule('/hello/', view_func=MyView.as_view('myview'))
if __name__ == '__main__':
app.run()
在类视图中通过methods属性设置当前类视图可以处理的请求方式。例如,在上述示例的MyView类中,设置当前类视图处理GET请求和POST请求,并在dispatch_request()方法中添加处理GET请求的具体逻辑。
from flask.views import View
class MyView(View): # 定义类视图
methods = ['GET', 'POST'] # 指定请求方式
def dispatch_request(self, name): # 重写dispatch_request()方法
if request.method == 'GET':
return f'hello {name}'
app = Flask(__name__)
# 将类视图与URL规则进行映射
app.add_url_rule('/hello/', view_func=MyView.as_view('myview'))
基于方法的类视图
在Flask中,基于方法的类视图需要继承flask.views模块中的MethodView类,而MethodView类继承View类
from flask.views import MethodView
由于MethodView类中已经重写了dispatch_request()方法,所以定义基于请求方法的类视图时不需要重写dispatch_request()方法。
- 在基于方法的类视图中,并非通过类属性methods来指定当前视图可以处理的请求方式,而是通过定义与请求方式同名的方法来处理不同的请求。例如,定义一个基于方法的类视图LoginView,之后在该类中添加处理GET请求和POST请求的方法。
from flask import Flask
from flask.views import MethodView
class LoginView(MethodView):
def get(self): # 处理GET请求
return '我负责处理GET请求'
def post(self): # 处理POST请求
return '我负责处理POST请求'
app = Flask(__name__)
#基于方法的类视图同样需要使用add_url_rule()方法将类视图与URL规则进行映射,并将通过as_view()方法将类视图转换后的视图函数传入view_func参数。
app.add_url_rule('/login', view_func=LoginView.as_view('login'))
if __name__ == '__main__':
app.run()
案例:
(1)在templates文件夹中添加一个用于展示用户登录页面的模板文件login.html
(2)在app.py文件中定义与使用基于方法的类视图
from flask.views import MethodView
from flask import Flask, render_template, request
class LoginView(MethodView):
def get(self): # 处理GET请求
return render_template('login.html')
def post(self): # 处理POST请求
username = request.form.get('username') # 获取输入的用户名
password = request.form.get('password') # 获取输入的密码
if username =='flask' and password == '123': # 判断用户名和密码是否为123
return f'用户:{username}登录成功。'
else:
return '用户名或密码错误,请重新登录。'
app = Flask(__name__)
app.add_url_rule('/login', view_func=LoginView.as_view('login'))
结果:
蓝图是一种制作应用程序组件的方式,可以在应用程序内部或跨越多个项目使用。当分配请求时,Flask会将蓝图和视图函数关联起来,并生成两个端点之间的URL。
- 若想在Flask程序中使用蓝图,首先需要创建蓝图,然后再对蓝图进行注册,其中创建蓝图需要通过Blueprint类实现;注册蓝图需要通过register_blueprint()方法实现。
flask.Blueprint(name, import_name, static_folder=None, static_url_path=None, template_folder=None, url_prefix=None, subdomain=None, url_defaults=None, root_path=None, cli_group=
•name:必选参数,表示蓝图的名称。
•import_name:必选参数,表示蓝图包的名称,通常为__name__。
•static_folder:可选参数,表示静态文件夹的路径。
•static_url_path:可选参数,表示静态文件的URL。
•template_folder:可选参数,表示模板文件夹路径。
•url_prefix:可选参数,表示附加到所有蓝图URL的路径,用于与Flask应用程序的其他URL区分。
-假设Flask程序包含4个视图函数,分别属于普通用户和管理员两个子模块。如果在该程序中使用蓝图,那么使用蓝图前后的程序结构分别如下图:
在项目的根目录下创建两个py文件,分别是user.py和admin.py,在这两个文件中分别创建user蓝图和admin蓝图
user.py
from flask import Blueprint
# 创建蓝图
user = Blueprint('user', __name__)
@user.route('/login')
def login():
return 'user_login'
@user.route('/register')
def register():
return 'user_register'
admin.py
from flask import Blueprint
# 创建蓝图
admin = Blueprint('admin', __name__)
@admin.route('/login')
def login():
return 'admin_login'
@admin.route('/add')
def add():
return 'admin_add'
- register_blueprint()方法用于将蓝图注册到Flask程序中。
register_blueprint(blueprint, url_prefix, subdomain, url_defaults,**options)
•blueprint:必选参数,表示要注册的蓝图。
•url_prefix:可选参数,表示附加到所有蓝图URL的路径,若在Blueprint类设置参url_prefix,则会被该参数值覆盖。
- 在项目的app.py文件中使用register_blueprint()方法注册user蓝图和admin蓝图。
from admin import admin
from user import user
from flask import Flask
app = Flask(__name__)
app.register_blueprint(admin, url_prefix='/admin') # 将蓝图admin进行注册
app.register_blueprint(user, url_prefix='/user') # 将蓝图user进行注册
if __name__ == '__main__':
app.run()
结果: