Flask 即插視圖基本原則

2021-08-23 18:37 更新

想象你有一個從數(shù)據(jù)庫載入一個對象列表并渲染到視圖的函數(shù):

@app.route('/users/')
def show_users(page):
    users = User.query.all()
    return render_template('users.html', users=users)

這是簡單而靈活的,但如果你想要用一種通用的,同樣可以適應其它模型和模板的 方式來提供這個視圖,你會需要更大的靈活性。這就是基于類的即插視圖所做的。 第一步,把它轉換為基于類的視圖,你要這樣做:

from flask.views import View

class ShowUsers(View):

    def dispatch_request(self):
        users = User.query.all()
        return render_template('users.html', objects=users)

app.add_url_rule('/users/', ShowUsers.as_view('show_users'))

如你所見,你需要做的是創(chuàng)建一個 flask.views.View 的子類, 并且實現(xiàn) dispatch_request() 。然后我們需要用類方法 as_view() 把這個類轉換到一個實際的視圖函數(shù)。你傳給 這個函數(shù)的字符串是視圖之后的最終名稱。但是用它自己實現(xiàn)的方法不夠有效,所以 我們稍微重構一下代碼:

from flask.views import View

class ListView(View):

    def get_template_name(self):
        raise NotImplementedError()

    def render_template(self, context):
        return render_template(self.get_template_name(), **context)

    def dispatch_request(self):
        context = {'objects': self.get_objects()}
        return self.render_template(context)

class UserView(ListView):

    def get_template_name(self):
        return 'users.html'

    def get_objects(self):
        return User.query.all()

這當然不是那么有助于一個小例子,但是對于解釋基本原則已經很有用了。當你有一 個基于類的視圖,那么問題來了, self 指向什么。它工作的方式是,無論何時請 求被調度,會創(chuàng)建這個類的一個新實例,并且 dispatch_request() 方法會以 URL 規(guī)則為參數(shù)調用。 這個類本身會用傳遞到 as_view() 函數(shù)的參數(shù)來實例化。 比如,你可以像這樣寫一個類:

class RenderTemplateView(View):
    def __init__(self, template_name):
        self.template_name = template_name
    def dispatch_request(self):
        return render_template(self.template_name)

然后你可以這樣注冊它:: And then you can register it like this:

app.add_url_rule('/about', view_func=RenderTemplateView.as_view(
    'about_page', template_name='about.html'))


以上內容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號