成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

underscore 系列之實(shí)現(xiàn)一個模板引擎(上)

LeexMuller / 638人閱讀

摘要:第一版我們來嘗試實(shí)現(xiàn)第一版第一版為了驗(yàn)證是否有用文件文件完整的可以查看示例一在這里我們使用了,實(shí)際上在文章中使用的是構(gòu)造函數(shù)。構(gòu)造函數(shù)創(chuàng)建一個新的對象。

前言

underscore 提供了模板引擎的功能,舉個例子:

var tpl = "hello: <%= name %>";

var compiled = _.template(tpl);
compiled({name: "Kevin"}); // "hello: Kevin"

感覺好像沒有什么強(qiáng)大的地方,再來舉個例子:

在 HTML 文件中:

    JavaScript 文件中:

    var container = document.getElementById("user_tmpl");
    
    var data = {
        users: [
            { "name": "Kevin", "url": "http://localhost" },
            { "name": "Daisy", "url": "http://localhost" },
            { "name": "Kelly", "url": "http://localhost" }
        ]
    }
    var precompile = _.template(document.getElementById("user_tmpl").innerHTML);
    var html = precompile(data);
    
    container.innerHTML = html;

    效果為:

    那么該如何實(shí)現(xiàn)這樣一個 _.template 函數(shù)呢?

    實(shí)現(xiàn)思路

    underscore 的 template 函數(shù)參考了 jQuery 的作者 John Resig 在 2008 年發(fā)表的一篇文章 JavaScript Micro-Templating,我們先從這篇文章的思路出發(fā),思考一下如何寫一個簡單的模板引擎。

    依然是以這段模板字符串為例:

    <%for ( var i = 0; i < users.length; i++ ) { %>
        
  • <%=users[i].name%>
  • <% } %>

    John Resig 的思路是將這段代碼轉(zhuǎn)換為這樣一段程序:

    // 模擬數(shù)據(jù)
    var users = [{"name": "Kevin", "url": "http://localhost"}];
    
    var p = [];
    for (var i = 0; i < users.length; i++) {
        p.push("
  • "); p.push(users[i].name); p.push("
  • "); } // 最后 join 一下就可以得到最終拼接好的模板字符串 console.log(p.join("")) //
  • Kevin
  • 我們注意,模板其實(shí)是一段字符串,我們怎么根據(jù)一段字符串生成一段代碼呢?很容易就想到用 eval,那我們就先用 eval 吧。

    然后我們會發(fā)現(xiàn),為了轉(zhuǎn)換成這樣一段代碼,我們需要將<%xxx%>轉(zhuǎn)換為 xxx,其實(shí)就是去掉包裹的符號,還要將 <%=xxx%>轉(zhuǎn)化成 p.push(xxx),這些都可以用正則實(shí)現(xiàn),但是我們還需要寫 p.push("

  • 、p.push("">");吶,這些該如何實(shí)現(xiàn)呢?

    那我們換個思路,依然是用正則,但是我們

    %> 替換成 p.push("

    <% 替換成 ");

    <%=xxx%> 替換成 ");p.push(xxx);p.push("

    我們來舉個例子:

    <%for ( var i = 0; i < users.length; i++ ) { %>
        
  • <%=users[i].name%>
  • <% } %>

    按照這個替換規(guī)則會被替換為:

    ");for ( var i = 0; i < users.length; i++ ) { p.push("
        
  • ");p.push(users[i].name);p.push("
  • "); } p.push("

    這樣肯定會報(bào)錯,畢竟代碼都沒有寫全,我們在首和尾加上部分代碼,變成:

    // 添加的首部代碼
    var p = []; p.push("
    
    ");for ( var i = 0; i < users.length; i++ ) { p.push("
        
  • ");p.push(users[i].name);p.push("
  • "); } p.push(" // 添加的尾部代碼 ");

    我們整理下這段代碼:

    var p = []; p.push("");
    for ( var i = 0; i < users.length; i++ ) { 
        p.push("
  • "); p.push(users[i].name); p.push("
  • "); } p.push("");

    恰好可以實(shí)現(xiàn)這個功能,不過還要注意一點(diǎn),要將換行符替換成空格,防止解析成代碼的時(shí)候報(bào)錯,不過在這里為了方便理解原理,就只在代碼里實(shí)現(xiàn)。

    第一版

    我們來嘗試實(shí)現(xiàn)第一版:

    // 第一版
    function tmpl(str, data) {
        var str = document.getElementById(str).innerHTML;
    
        var string = "var p = []; p.push("" +
        str
        .replace(/[
    	
    ]/g, "")
        .replace(/<%=(.*?)%>/g, "");p.push($1);p.push("")
        .replace(/<%/g, "");")
        .replace(/%>/g,"p.push("")
        + "");"
    
        eval(string)
    
        return p.join("");
    };

    為了驗(yàn)證是否有用:

    HTML 文件:

    JavaScript 文件:

    var users = [
        { "name": "Byron", "url": "http://localhost" },
        { "name": "Casper", "url": "http://localhost" },
        { "name": "Frank", "url": "http://localhost" }
    ]
    tmpl("user_tmpl", users)

    完整的 Demo 可以查看 template 示例一

    Function

    在這里我們使用了 eval ,實(shí)際上 John Resig 在文章中使用的是 Function 構(gòu)造函數(shù)。

    Function 構(gòu)造函數(shù)創(chuàng)建一個新的 Function 對象。 在 JavaScript 中, 每個函數(shù)實(shí)際上都是一個 Function 對象。

    使用方法為:

    new Function ([arg1[, arg2[, ...argN]],] functionBody)

    arg1, arg2, ... argN 表示函數(shù)用到的參數(shù),functionBody 表示一個含有包括函數(shù)定義的 JavaScript 語句的字符串。

    舉個例子:

    var adder = new Function("a", "b", "return a + b");
    
    adder(2, 6); // 8

    那么 John Resig 到底是如何實(shí)現(xiàn)的呢?

    第二版

    使用 Function 構(gòu)造函數(shù):

    // 第二版
    function tmpl(str, data) {
        var str = document.getElementById(str).innerHTML;
    
        var fn = new Function("obj",
    
        "var p = []; p.push("" +
    
        str
        .replace(/[
    	
    ]/g, "")
        .replace(/<%=(.*?)%>/g, "");p.push($1);p.push("")
        .replace(/<%/g, "");")
        .replace(/%>/g,"p.push("")
        + "");return p.join("");");
    
        return fn(data);
    };

    使用方法依然跟第一版相同,具體 Demo 可以查看 template 示例二

    不過值得注意的是:其實(shí) tmpl 函數(shù)沒有必要傳入 data 參數(shù),也沒有必要在最后 return 的時(shí)候,傳入 data 參數(shù),即使你把這兩個參數(shù)都去掉,代碼還是可以正常執(zhí)行的。

    這是因?yàn)?

    使用Function構(gòu)造器生成的函數(shù),并不會在創(chuàng)建它們的上下文中創(chuàng)建閉包;它們一般在全局作用域中被創(chuàng)建。當(dāng)運(yùn)行這些函數(shù)的時(shí)候,它們只能訪問自己的本地變量和全局變量,不能訪問Function構(gòu)造器被調(diào)用生成的上下文的作用域。這和使用帶有函數(shù)表達(dá)式代碼的 eval 不同。

    這里之所以依然傳入了 data 參數(shù),是為了下一版做準(zhǔn)備。

    with

    現(xiàn)在有一個小問題,就是實(shí)際上我們傳入的數(shù)據(jù)結(jié)構(gòu)可能比較復(fù)雜,比如:

    var data = {
        status: 200,
        name: "kevin",
        friends: [...]
    }

    如果我們將這個數(shù)據(jù)結(jié)構(gòu)傳入 tmpl 函數(shù)中,在模板字符串中,如果要用到某個數(shù)據(jù),總是需要使用 data.name、data.friends 的形式來獲取,麻煩就麻煩在我想直接使用 name、friends 等變量,而不是繁瑣的使用 data. 來獲取。

    這又該如何實(shí)現(xiàn)的呢?答案是 with。

    with 語句可以擴(kuò)展一個語句的作用域鏈(scope chain)。當(dāng)需要多次訪問一個對象的時(shí)候,可以使用 with 做簡化。比如:

    var hostName = location.hostname;
    var url = location.href;
    
    // 使用 with
    with(location){
        var hostname = hostname;
        var url = href;
    }
    function Person(){
        this.name = "Kevin";
        this.age = "18";
    }
    
    var person = new Person();
    
    with(person) {
        console.log("my name is " + name + ", age is " + age + ".")
    }
    // my name is Kevin, age is 18.

    最后:不建議使用 with 語句,因?yàn)樗赡苁腔煜e誤和兼容性問題的根源,除此之外,也會造成性能低下

    第三版

    使用 with ,我們再寫一版代碼:

    // 第三版
    function tmpl(str, data) {
        var str = document.getElementById(str).innerHTML;
    
        var fn = new Function("obj",
    
        // 其實(shí)就是這里多添加了一句 with(obj){...}
        "var p = []; with(obj){p.push("" +
    
        str
        .replace(/[
    	
    ]/g, "")
        .replace(/<%=(.*?)%>/g, "");p.push($1);p.push("")
        .replace(/<%/g, "");")
        .replace(/%>/g,"p.push("")
        + "");}return p.join("");");
    
        return fn(data);
    };

    具體 Demo 可以查看 template 示例三

    第四版

    如果我們的模板不變,數(shù)據(jù)卻發(fā)生了變化,如果使用我們的之前寫的 tmpl 函數(shù),每次都會 new Function,這其實(shí)是沒有必要的,如果我們能在使用 tmpl 的時(shí)候,返回一個函數(shù),然后使用該函數(shù),傳入不同的數(shù)據(jù),只根據(jù)數(shù)據(jù)不同渲染不同的 html 字符串,就可以避免這種無謂的損失。

    // 第四版
    function tmpl(str, data) {
        var str = document.getElementById(str).innerHTML;
    
        var fn = new Function("obj",
    
        "var p = []; with(obj){p.push("" +
    
        str
        .replace(/[
    	
    ]/g, "")
        .replace(/<%=(.*?)%>/g, "");p.push($1);p.push("")
        .replace(/<%/g, "");")
        .replace(/%>/g,"p.push("")
        + "");}return p.join("");");
    
        var template = function(data) {
            return fn.call(this, data)
        }
        return template;
    };
    
    // 使用時(shí)
    var compiled = tmpl("user_tmpl");
    results.innerHTML = compiled(data);

    具體 Demo 可以查看 template 示例四

    下期預(yù)告

    至此,我們已經(jīng)跟著 jQuery 的作者 John Resig 實(shí)現(xiàn)了一個簡單的模板引擎,雖然 underscore 基于這個思路實(shí)現(xiàn),但是功能強(qiáng)大,相對的,代碼也更加復(fù)雜一下,下一篇,我們一起去分析 underscore 的 template 函數(shù)實(shí)現(xiàn)。

    underscore 系列

    underscore 系列目錄地址:https://github.com/mqyqingfeng/Blog。

    underscore 系列預(yù)計(jì)寫八篇左右,重點(diǎn)介紹 underscore 中的代碼架構(gòu)、鏈?zhǔn)秸{(diào)用、內(nèi)部函數(shù)、模板引擎等內(nèi)容,旨在幫助大家閱讀源碼,以及寫出自己的 undercore。

    如果有錯誤或者不嚴(yán)謹(jǐn)?shù)牡胤剑垊?wù)必給予指正,十分感謝。如果喜歡或者有所啟發(fā),歡迎 star,對作者也是一種鼓勵。

  • 文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

    轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/92271.html

    相關(guān)文章

    • underscore 的源碼該如何閱讀?

      摘要:所以它與其他系列的文章并不沖突,完全可以在閱讀完這個系列后,再跟著其他系列的文章接著學(xué)習(xí)。如何閱讀我在寫系列的時(shí)候,被問的最多的問題就是該怎么閱讀源碼我想簡單聊一下自己的思路。感謝大家的閱讀和支持,我是冴羽,下個系列再見啦 前言 別名:《underscore 系列 8 篇正式完結(jié)!》 介紹 underscore 系列是我寫的第三個系列,前兩個系列分別是 JavaScript 深入系列、...

      weknow619 評論0 收藏0
    • underscore 系列實(shí)現(xiàn)一個模板引擎(下)

      摘要:前言本篇接著上篇系列之實(shí)現(xiàn)一個模板引擎上。字符串中的每個字符均可由一個轉(zhuǎn)義序列表示。在中,有四個字符被認(rèn)為是行終結(jié)符,其他的折行字符都會被視為空白。 前言 本篇接著上篇 underscore 系列之實(shí)現(xiàn)一個模板引擎(上)。 鑒于本篇涉及的知識點(diǎn)太多,我們先來介紹下會用到的知識點(diǎn)。 反斜杠的作用 var txt = We are the so-called Vikings from th...

      gyl_coder 評論0 收藏0
    • 淺談 Web 中前后端模板引擎的使用

      摘要:前端模板的出現(xiàn)使得前后端分離成為可能??偨Y(jié)本文簡單介紹了模板引擎在前后端的使用,下文我們回到,重點(diǎn)分析下的使用方式以及源碼原理。樓主對于模板引擎的認(rèn)識比較淺顯,有不正之處希望指出感謝 前言 這篇文章本來不打算寫的,實(shí)話說樓主對前端模板的認(rèn)識還處在非常初級的階段,但是為了整個 源碼解讀系列 的完整性,在深入 Underscore _.template 方法源碼后,覺得還是有必要記下此文,...

      chenjiang3 評論0 收藏0
    • underscore 系列內(nèi)部函數(shù) restArgs

      摘要:與最后,使用我們的寫的函數(shù)重寫下函數(shù)系列系列目錄地址。系列預(yù)計(jì)寫八篇左右,重點(diǎn)介紹中的代碼架構(gòu)鏈?zhǔn)秸{(diào)用內(nèi)部函數(shù)模板引擎等內(nèi)容,旨在幫助大家閱讀源碼,以及寫出自己的。如果有錯誤或者不嚴(yán)謹(jǐn)?shù)牡胤?,請?wù)必給予指正,十分感謝。 partial 在《 JavaScript 專題之偏函數(shù)》中,我們寫了一個 partial 函數(shù),用來固定函數(shù)的部分參數(shù),實(shí)現(xiàn)代碼如下: // 這是文章中的第一版 fu...

      zzzmh 評論0 收藏0
    • underscore 系列字符實(shí)體與 _.escape

      摘要:前言提供了函數(shù),用于轉(zhuǎn)義字符串,替換和字符為字符實(shí)體。如果希望正確地顯示預(yù)留字符,我們必須在源代碼中使用字符實(shí)體。字符實(shí)體有兩種形式。轉(zhuǎn)義我們的應(yīng)對方式就是將取得的值中的特殊字符轉(zhuǎn)為字符實(shí)體。 前言 underscore 提供了 _.escape 函數(shù),用于轉(zhuǎn)義 HTML 字符串,替換 &, , , , 和 ` 字符為字符實(shí)體。 _.escape(Curly, Larry & Moe)...

      only_do 評論0 收藏0

    發(fā)表評論

    0條評論

    LeexMuller

    |高級講師

    TA的文章

    閱讀更多
    最新活動
    閱讀需要支付1元查看
    <