黄色网页视频 I 影音先锋日日狠狠久久 I 秋霞午夜毛片 I 秋霞一二三区 I 国产成人片无码视频 I 国产 精品 自在自线 I av免费观看网站 I 日本精品久久久久中文字幕5 I 91看视频 I 看全色黄大色黄女片18 I 精品不卡一区 I 亚洲最新精品 I 欧美 激情 在线 I 人妻少妇精品久久 I 国产99视频精品免费专区 I 欧美影院 I 欧美精品在欧美一区二区少妇 I av大片网站 I 国产精品黄色片 I 888久久 I 狠狠干最新 I 看看黄色一级片 I 黄色精品久久 I 三级av在线 I 69色综合 I 国产日韩欧美91 I 亚洲精品偷拍 I 激情小说亚洲图片 I 久久国产视频精品 I 国产综合精品一区二区三区 I 色婷婷国产 I 最新成人av在线 I 国产私拍精品 I 日韩成人影音 I 日日夜夜天天综合

詳解Python的Django框架中的模版繼承

系統(tǒng) 1902 0

在實(shí)際應(yīng)用中,你將用 Django 模板系統(tǒng)來(lái)創(chuàng)建整個(gè) HTML 頁(yè)面。 這就帶來(lái)一個(gè)常見(jiàn)的 Web 開(kāi)發(fā)問(wèn)題: 在整個(gè)網(wǎng)站中,如何減少共用頁(yè)面區(qū)域(比如站點(diǎn)導(dǎo)航)所引起的重復(fù)和冗余代碼?

解決該問(wèn)題的傳統(tǒng)做法是使用 服務(wù)器端的 includes ,你可以在 HTML 頁(yè)面中使用該指令將一個(gè)網(wǎng)頁(yè)嵌入到另一個(gè)中。 事實(shí)上, Django 通過(guò)剛才講述的 {% include %} 支持了這種方法。 但是用 Django 解決此類(lèi)問(wèn)題的首選方法是使用更加優(yōu)雅的策略―― 模板繼承 。

本質(zhì)上來(lái)說(shuō),模板繼承就是先構(gòu)造一個(gè)基礎(chǔ)框架模板,而后在其子模板中對(duì)它所包含站點(diǎn)公用部分和定義塊進(jìn)行重載。

讓我們通過(guò)修改 current_datetime.html 文件,為 current_datetime 創(chuàng)建一個(gè)更加完整的模板來(lái)體會(huì)一下這種做法:

            
              The current time
            
            

My helpful timestamp site

It is now {{ current_date }}.


Thanks for visiting my site.

這看起來(lái)很棒,但如果我們要為第三章的 hours_ahead 視圖創(chuàng)建另一個(gè)模板會(huì)發(fā)生什么事情呢?

              
                Future time
              
              

My helpful timestamp site

In {{ hour_offset }} hour(s), it will be {{ next_time }}.


Thanks for visiting my site.

很明顯,我們剛才重復(fù)了大量的 HTML 代碼。 想象一下,如果有一個(gè)更典型的網(wǎng)站,它有導(dǎo)航條、樣式表,可能還有一些 JavaScript 代碼,事情必將以向每個(gè)模板填充各種冗余的 HTML 而告終。

解決這個(gè)問(wèn)題的服務(wù)器端 include 方案是找出兩個(gè)模板中的共同部分,將其保存為不同的模板片段,然后在每個(gè)模板中進(jìn)行 include。 也許你會(huì)把模板頭部的一些代碼保存為 header.html 文件:

          

你可能會(huì)把底部保存到文件 footer.html :

            

Thanks for visiting my site.

對(duì)基于 include 的策略,頭部和底部的包含很簡(jiǎn)單。 麻煩的是中間部分。 在此范例中,每個(gè)頁(yè)面都有一個(gè)

My helpful timestamp site

標(biāo)題,但是這個(gè)標(biāo)題不能放在 header.html 中,因?yàn)槊總€(gè)頁(yè)面的 是不同的。 如果我們將 <h1> 包含在頭部,我們就不得不包含 <title> ,但這樣又不允許在每個(gè)頁(yè)面對(duì)它進(jìn)行定制。 何去何從呢?</p> <p>Django 的模板繼承系統(tǒng)解決了這些問(wèn)題。 你可以將其視為服務(wù)器端 include 的逆向思維版本。 你可以對(duì)那些 不同 的代碼段進(jìn)行定義,而不是 共同 代碼段。</p> <p>第一步是定義 基礎(chǔ)模板 , 該框架之后將由 子模板 所繼承。 以下是我們目前所講述范例的基礎(chǔ)模板:</p> <div class="jb51code"> <pre class="brush:py;"> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN"> <html lang="en"> <head> <title>{% block title %}{% endblock %}

My helpful timestamp site

{% block content %}{% endblock %} {% block footer %}

Thanks for visiting my site.

{% endblock %}

這個(gè)叫做 base.html 的模板定義了一個(gè)簡(jiǎn)單的 HTML 框架文檔,我們將在本站點(diǎn)的所有頁(yè)面中使用。 子模板的作用就是重載、添加或保留那些塊的內(nèi)容。 (如果你一直按順序?qū)W習(xí)到這里,保存這個(gè)文件到你的template目錄下,命名為 base.html .)

我們使用一個(gè)以前已經(jīng)見(jiàn)過(guò)的模板標(biāo)簽: {% block %} 。 所有的 {% block %} 標(biāo)簽告訴模板引擎,子模板可以重載這些部分。 每個(gè){% block %}標(biāo)簽所要做的是告訴模板引擎,該模板下的這一塊內(nèi)容將有可能被子模板覆蓋。

現(xiàn)在我們已經(jīng)有了一個(gè)基本模板,我們可以修改 current_datetime.html 模板來(lái) 使用它:

          
{% extends "base.html" %}

{% block title %}The current time{% endblock %}

{% block content %}

          

It is now {{ current_date }}.

{% endblock %}

再為 hours_ahead 視圖創(chuàng)建一個(gè)模板,看起來(lái)是這樣的:

          
{% extends "base.html" %}

{% block title %}Future time{% endblock %}

{% block content %}

          

In {{ hour_offset }} hour(s), it will be {{ next_time }}.

{% endblock %}

看起來(lái)很漂亮是不是? 每個(gè)模板只包含對(duì)自己而言 獨(dú)一無(wú)二 的代碼。 無(wú)需多余的部分。 如果想進(jìn)行站點(diǎn)級(jí)的設(shè)計(jì)修改,僅需修改 base.html ,所有其它模板會(huì)立即反映出所作修改。

以下是其工作方式。 在加載 current_datetime.html 模板時(shí),模板引擎發(fā)現(xiàn)了 {% extends %} 標(biāo)簽, 注意到該模板是一個(gè)子模板。 模板引擎立即裝載其父模板,即本例中的 base.html 。

此時(shí),模板引擎注意到 base.html 中的三個(gè) {% block %} 標(biāo)簽,并用子模板的內(nèi)容替換這些 block 。因此,引擎將會(huì)使用我們?cè)?{ block title %} 中定義的標(biāo)題,對(duì) {% block content %} 也是如此。 所以,網(wǎng)頁(yè)標(biāo)題一塊將由 {% block title %}替換,同樣地,網(wǎng)頁(yè)的內(nèi)容一塊將由 {% block content %}替換。

注意由于子模板并沒(méi)有定義 footer 塊,模板系統(tǒng)將使用在父模板中定義的值。 父模板 {% block %} 標(biāo)簽中的內(nèi)容總是被當(dāng)作一條退路。

繼承并不會(huì)影響到模板的上下文。 換句話說(shuō),任何處在繼承樹(shù)上的模板都可以訪問(wèn)到你傳到模板中的每一個(gè)模板變量。

你可以根據(jù)需要使用任意多的繼承次數(shù)。 使用繼承的一種常見(jiàn)方式是下面的三層法:

  • ??? 創(chuàng)建 base.html 模板,在其中定義站點(diǎn)的主要外觀感受。 這些都是不常修改甚至從不修改的部分。
  • ??? 為網(wǎng)站的每個(gè)區(qū)域創(chuàng)建 base_SECTION.html 模板(例如, base_photos.html 和 base_forum.html )。這些模板對(duì) base.html 進(jìn)行拓展,并包含區(qū)域特定的風(fēng)格與設(shè)計(jì)。
  • ??? 為每種類(lèi)型的頁(yè)面創(chuàng)建獨(dú)立的模板,例如論壇頁(yè)面或者圖片庫(kù)。 這些模板拓展相應(yīng)的區(qū)域模板。

這個(gè)方法可最大限度地重用代碼,并使得向公共區(qū)域(如區(qū)域級(jí)的導(dǎo)航)添加內(nèi)容成為一件輕松的工作。

以下是使用模板繼承的一些訣竅:

  • ??? 如果在模板中使用 {% extends %} ,必須保證其為模板中的第一個(gè)模板標(biāo)記。 否則,模板繼承將不起作用。
  • ??? 一般來(lái)說(shuō),基礎(chǔ)模板中的 {% block %} 標(biāo)簽越多越好。 記住,子模板不必定義父模板中所有的代碼塊,因此你可以用合理的缺省值對(duì)一些代碼塊進(jìn)行填充,然后只對(duì)子模板所需的代碼塊進(jìn)行(重)定義。 俗話說(shuō),鉤子越多越好。
  • ??? 如果發(fā)覺(jué)自己在多個(gè)模板之間拷貝代碼,你應(yīng)該考慮將該代碼段放置到父模板的某個(gè) {% block %} 中。
  • ??? 如果你需要訪問(wèn)父模板中的塊的內(nèi)容,使用 {{ block.super }}這個(gè)標(biāo)簽吧,這一個(gè)魔法變量將會(huì)表現(xiàn)出父模板中的內(nèi)容。 如果只想在上級(jí)代碼塊基礎(chǔ)上添加內(nèi)容,而不是全部重載,該變量就顯得非常有用了。
  • ??? 不允許在同一個(gè)模板中定義多個(gè)同名的 {% block %} 。 存在這樣的限制是因?yàn)閎lock 標(biāo)簽的工作方式是雙向的。 也就是說(shuō),block 標(biāo)簽不僅挖了一個(gè)要填的坑,也定義了在父模板中這個(gè)坑所填充的內(nèi)容。如果模板中出現(xiàn)了兩個(gè)相同名稱(chēng)的 {% block %} 標(biāo)簽,父模板將無(wú)從得知要使用哪個(gè)塊的內(nèi)容。
  • ??? {% extends %} 對(duì)所傳入模板名稱(chēng)使用的加載方法和 get_template() 相同。 也就是說(shuō),會(huì)將模板名稱(chēng)被添加到 TEMPLATE_DIRS 設(shè)置之后。
  • ??? 多數(shù)情況下, {% extends %} 的參數(shù)應(yīng)該是字符串,但是如果直到運(yùn)行時(shí)方能確定父模板名,這個(gè)參數(shù)也可以是個(gè)變量。 這使得你能夠?qū)崿F(xiàn)一些很酷的動(dòng)態(tài)功能。


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號(hào)聯(lián)系: 360901061

您的支持是博主寫(xiě)作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對(duì)您有幫助就好】

您的支持是博主寫(xiě)作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長(zhǎng)會(huì)非常 感謝您的哦?。。?/p>

發(fā)表我的評(píng)論
最新評(píng)論 總共0條評(píng)論