之前對bottle做過不少的介紹,也寫過一些文章來說明bottle的缺點,最近發(fā)現(xiàn)其實之前有些地方說的不太公平,所以趁此機會也來更正一下。
bottle是支持類似flask url_for的語法的,具體使用方法在下文介紹
bottle的request.query之類的參數默認是str類型,也是有原因的,比如我在給google做代理的時候,編碼就不一定是utf8的,如果強制轉化utf8就會報錯
之前的bug也得到了修正,比如mount(‘/x',app)之后,/x/和/x都可以訪問到
OK,現(xiàn)在正式進入主題,我們來介紹一些bottle的一些高級使用
一. 智能創(chuàng)建url
這部分在bottle的文檔上是沒有介紹的(其實bottle明明實現(xiàn)了很多貼心的功能,不知道為啥都不寫在文檔上)。
在Bottle類里,有一個成員函數:
def get_url(self, routename, **kargs): """ Return a string that matches a named route """ scriptname = request.environ.get('SCRIPT_NAME', '').strip('/') + '/' location = self.router.build(routename, **kargs).lstrip('/') return urljoin(urljoin('/', scriptname), location) def get_url(self, routename, **kargs): """ Return a string that matches a named route """ scriptname = request.environ.get('SCRIPT_NAME', '').strip('/') + '/' location = self.router.build(routename, **kargs).lstrip('/') return urljoin(urljoin('/', scriptname), location)
那么這個routename是哪里來的呢?看 route 裝飾器的參數:
def route(self, path=None, method='GET', callback=None, name=None, apply=None, skip=None, **config): def route(self, path=None, method='GET', callback=None, name=None, apply=None, skip=None, **config):
其中的name參數就是routename(這里不得不說一下,這種方式比flask要好些,要用才指定name,而不需要為了實現(xiàn)url_for,把整個框架都實現(xiàn)的很復雜)
所以看到這里大家也就明白了,bottle的url生成器是綁定在Bottle實例上的,所以跨實例訪問默認是做不到的。
而可能由于bottle所推崇的micro化,所以其源碼中特意對默認Bottle示例包裝出了一個函數:
for name in '''route get post put delete error mount hook install uninstall'''.split(): globals()[name] = make_default_app_wrapper(name)url = make_default_app_wrapper('get_url')del name for name in '''route get post put delete error mount hook install uninstall'''.split(): globals()[name] = make_default_app_wrapper(name)url = make_default_app_wrapper('get_url')del name
這樣做的好處是,如果工程只用到默認的Bottle實例的話,在模板中就可以直接使用url,而不必再多傳個Bottle實例進去。
更正一下,bottle的get_url是不能跨app調用的,比如被mount的app調用主app的get_url會錯掉,因為此時的SCRIPT_NAME是當前頁的path,所以拼裝起來會亂掉,所以就不要嘗試了。
新聞熱點
疑難解答