一個不成熟的模板引擎思路

關(guān)于模板,我倒是用過了不少冯事。最開始要數(shù)Java的JSP了宿接,然后接觸了PHP的smarty赘淮,再就是Python的jinja2, Django內(nèi)置模板,現(xiàn)在剛開始看Nodejs睦霎,也發(fā)現(xiàn)了不少類似的模板引擎梢卸,ejs, jade等等吧。

模板帶來的最直接的好處就是加速開發(fā)副女,前后端分離蛤高。除此之外,對于字符串的格式化同樣是個比較好的應(yīng)用。習慣了Python中

string = "hello {}".format("郭璞")  # hello 郭璞
string = "hello {username}".format(username="郭璞") # hello 郭璞

這樣簡便的用法戴陡,突然來到nodejs中塞绿,沒有了這類特性的原生支持,寫起來打印語句就老是覺得很別扭恤批,一點都不優(yōu)雅异吻。然后我就想自己做一個實現(xiàn)上述功能的工具函數(shù),方便自己的使用开皿。然后就想到了模板這一個方向涧黄,雖然想法還不夠成熟,甚至是有點拙略赋荆,但是“靈(瞎)感(鬧)”還是得記錄一下不是笋妥。


Function對象

JavaScript中有這么一個神奇的對象,那就是Function窄潭。如果函數(shù)體符合語法要求春宣,那么你就可以動態(tài)創(chuàng)建出一個自己的函數(shù)出來。下面來個簡單的小例子嫉你。

無參模式

function create_function(){
    var func_body = "var time = new Date(); console.log('創(chuàng)建時間:'+time);";
    var func = new Function('', func_body);
    func();
}
create_function();

運行結(jié)果如下:

E:\Code\Nodejs\learn\my-work\string>node one.js
創(chuàng)建時間:Tue Jun 13 2017 15:40:15 GMT+0800 (中國標準時間)

E:\Code\Nodejs\learn\my-work\string>

有參模式

剛才演示了一個無參數(shù)的情況月帝,那么有參數(shù)的情況如何呢?

function create_function_with_parameters() {
    var param1 = "郭璞";
    var param2 = "遼寧大連";
    var func_body = "console.log('Hello '+param1+', welcome to '+param2+'!' );";
    var func = new Function('param1', 'param2', func_body);
    func(param1, param2);
}
create_function_with_parameters();

同樣的運行結(jié)果如下:

E:\Code\Nodejs\learn\my-work\string>node one.js
Hello 郭璞, welcome to 遼寧大連!

E:\Code\Nodejs\learn\my-work\string>

到這里幽污,關(guān)于Function的內(nèi)容就算是鋪墊完成了嚷辅。只需要了解這


正則

探究模板的真實原理,有些語言中是編譯型的距误,有些是替換型的簸搞。但是不管是哪種類型,都離不開出變量關(guān)鍵字這個步驟准潭。而這個過程用正則表達式基本上是最好的方法了趁俊。所以需要掌握一點相關(guān)的技巧。

如何表達刑然?

在Nodejs中寺擂,使用正則表達式有兩種形式:

  • 字面量: /pattern/flags
  • RegExp: new RegExp(pattern, flags)

關(guān)于正則表達式的具體的規(guī)則,鑒于篇幅很長泼掠,這里就不再贅述了怔软。有興趣的可以瀏覽下面的這篇文章。
http://www.cnblogs.com/chenmeng0818/p/6370819.html

需求獲取

根據(jù)一開始的設(shè)想择镇,目標是獲取{{}}{%%} 這種語法下的變量名稱爽雄,然后替換成對應(yīng)的變量值。 因此可以寫出如下的正則表達式:

var pattern1 = /{{([\s\S]+?)}}/gi;
// 或者
var pattern2 = /{%([\s\S]+?)%}/gi;

默認規(guī)則如下:

  • {{}} 中直接替換為變量名對應(yīng)的值沐鼠。
  • {%%} 中的則是可以添加到函數(shù)體的代碼塊挚瘟,要保留起來叹谁。

簡易實現(xiàn)

下面簡單的對照著實現(xiàn)一下。

直接變量形式

function test1(){
    var tpl = "Hello {{visitorname}}, Welcome to {{worldname}}!";
    var data = {
        visitorname: "游客",
        worldname: "冰雹工作室"
    };
    var pattern = /{{([\s\S]+?)}}/gi;
    var result = tpl.replace(pattern, (match, tuple)=>{
        return data[tuple];
    });

    console.log("渲染后的數(shù)據(jù)為:\n", result);
}

實現(xiàn)結(jié)果:

E:\Code\Nodejs\learn\my-work\string>node one.js
渲染后的數(shù)據(jù)為:
 Hello 游客, Welcome to 冰雹工作室!

E:\Code\Nodejs\learn\my-work\string>

對象形式

function test2(){
    var tpl = "I'm {{user.name}}, and I come from {{user.address}}";
    var user = {name: "郭璞", address: "遼寧大連"};
    console.log(user.name);
    var pattern = /{{([\s\S]+?)}}/gi;
    var result = tpl.replace(pattern, function(match, tuple, offset){
       return eval(''+tuple);
    });
    console.log(result);

}

運行效果:

E:\Code\Nodejs\learn\my-work\string>node one.js
郭璞
I'm 郭璞, and I come from 遼寧大連

E:\Code\Nodejs\learn\my-work\string>

混雜多參數(shù)實現(xiàn)

剛才實現(xiàn)了只有關(guān)鍵字的和有對象性質(zhì)的參數(shù)的例子乘盖,但是實際中情況可能比這要復(fù)雜的多焰檩,比如混雜模式。接下來著手實現(xiàn)一下混雜模式下的替換策略订框。

function test3(){
    var tpl = "I am {} of {} years old, and I come from {user.address}.";
    var name = '郭璞';
    var index = 0;
    var paramindex = 0;
    // var parameters = [{name: '郭璞'}, {'age': 22}, {address: '遼寧大連'}];
    var parameters = ['郭璞', 22, {user: {address: '遼寧大連'}}];
    console.log(parameters[2]);
    var result = tpl.replace(/{([\s\S])*?}/gi, function(match, tuple, offset){
        console.log('match:', match);
        console.log('tuple: ', tuple);
        tpl = tpl.slice(index, offset);
        index = offset + match.length;
        paramindex += 1;

        var temp = parameters[paramindex-1];
        if(match.length > 2){
            // 使用tuple不能正確獲取到標記中相關(guān)的變量名析苫,故用match來代替.
            match = match.slice(1, match.length-1);
            return eval('parameters[paramindex-1].'+match);
        }else{
            return temp;
        }
        // return parameters[paramindex-1];
    });
    console.log(result);
}

運行結(jié)果如下:

E:\Code\Nodejs\learn\my-work\string>node one.js
{ user: { address: '遼寧大連' } }
match: {}
tuple:  undefined
match: {}
tuple:  undefined
match: {user.address}
tuple:  s
******* s
I am 郭璞 of 22 years old, and I come from 遼寧大連.

E:\Code\Nodejs\learn\my-work\string>

關(guān)于正則這塊,大致的內(nèi)容就是這樣了穿扳。如果要想更簡單的調(diào)用衩侥,只需要封裝起來,用外部參數(shù)代替就好了矛物。

當然茫死,注意變量名的命名風格。


實戰(zhàn)

廢話連篇說了兩個小節(jié)履羞,還沒到正式的模板制作峦萎。下面就整合一下剛才例子。模擬著實現(xiàn)一下好了忆首。

(!完整)代碼

來個不完整的代碼爱榔,示意一下算了。

/**
 * 通過正則表達式和Function語法創(chuàng)建一個簡單的模板引擎糙及。
 */

const pattern = /{{([\s\S]+?)}}|{%([\s\S]+?)%}|$/img;

function template(text, params, name) {
    // 聲明最終要返回的解析好的文本串,也就是構(gòu)造Function所需的函數(shù)體部分详幽。
    var func_body = '';
    // 函數(shù)體里面最終效果是返回一個代表了解析完成的字符串的變量,因此要聲明一個出來
    func_body += 'var parsedstr="";';
    func_body += 'parsedstr+="';
    // 設(shè)置一個定位器浸锨,每次更新偏移量妒潭,進行全文替換工作
    var index = 0;
    // 開始正則匹配,根據(jù)捕獲到的元組進行剖析
    text.replace(pattern, function (matchedtext, interpolate, evaluate, offset) {
        // 匹配到正常的HTML文本揣钦,則直接添加到func_body中即可
        func_body += text.slice(index, offset);

        // 如果是evaluate類型的文本,則作為代碼進行拼接
        if (evaluate) {
            func_body += '";' + evaluate + 'parsedstr+="';
        }

        // 匹配到interpolate類型的文本漠酿,則作為變量值進行替換
        if (interpolate) {
            func_body += '"+' + interpolate + '+"';
        }

        // 更新偏移量index冯凹,讓程序向后移動
        index = offset + matchedtext.length;
        // 貌似返回值沒什么用吧
        return matchedtext;
    });

    // 完成函數(shù)體的構(gòu)建之后就可以調(diào)用Function的語法實現(xiàn)渲染函數(shù)的構(gòu)建了
    func_body += '"; return parsedstr;';

    return new Function('obj', 'name', func_body)(params, name);
}

function test() {
    var obj = [
        { text: '張三' },
        { text: '李四' },
        { text: '王五' },
        { text: '趙六' },
        { text: '韓七' },
        { text: '王八' }
    ];
    var name = '郭璞';


    var fs = require('fs');
    // var rawtext = fs.readFileSync('index.html').toString('utf8');
    var rawtext = '<ul>{%for(var i in obj){%}<li>{{ obj[i].text }}</li><br>{%}%}</ul>';
    console.log("源文件:", rawtext);
    var result = template(rawtext, obj);
    console.log("渲染后文件:", result, name);
    fs.writeFileSync('rendered.html', result);
    console.log('渲染完畢,請查看rendered.html文件')
}

test();

同級目錄下生成的文件內(nèi)容為:

<ul>
    <li>張三</li><br>
    <li>李四</li><br>
    <li>王五</li><br>
    <li>趙六</li><br>
    <li>韓七</li><br>
    <li>王八</li><br></ul>

感覺效果還行炒嘲,但是這里面參數(shù)太固定化了宇姚,實際封裝的時候還需要酌情指定,不然這東西也就沒什么卵用夫凸。

總結(jié)

要是論實用性價值的話浑劳,這個不成熟的模板實現(xiàn)思路毫無價值。但是對于我而言夭拌,用來格式化字符串倒是個不錯的選擇魔熏,估計我會把這個小思路封裝成一個小小的模塊衷咽,詳情可以關(guān)注我的GitHub的這個鏈接。

https://github.com/guoruibiao/have-fun-in-node

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蒜绽,一起剝皮案震驚了整個濱河市镶骗,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌躲雅,老刑警劉巖鼎姊,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異相赁,居然都是意外死亡相寇,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進店門钮科,熙熙樓的掌柜王于貴愁眉苦臉地迎上來唤衫,“玉大人,你說我怎么就攤上這事跺嗽≌绞冢” “怎么了?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵桨嫁,是天一觀的道長植兰。 經(jīng)常有香客問我,道長璃吧,這世上最難降的妖魔是什么楣导? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮畜挨,結(jié)果婚禮上筒繁,老公的妹妹穿的比我還像新娘。我一直安慰自己巴元,他們只是感情好毡咏,可當我...
    茶點故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著逮刨,像睡著了一般呕缭。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上修己,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天恢总,我揣著相機與錄音,去河邊找鬼睬愤。 笑死片仿,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的尤辱。 我是一名探鬼主播砂豌,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼厢岂,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了奸鸯?” 一聲冷哼從身側(cè)響起咪笑,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎娄涩,沒想到半個月后窗怒,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡蓄拣,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年扬虚,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片球恤。...
    茶點故事閱讀 38,724評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡辜昵,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出咽斧,到底是詐尸還是另有隱情堪置,我是刑警寧澤,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布张惹,位于F島的核電站舀锨,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏宛逗。R本人自食惡果不足惜坎匿,卻給世界環(huán)境...
    茶點故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望雷激。 院中可真熱鬧替蔬,春花似錦、人聲如沸屎暇。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽根悼。三九已至凶异,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間番挺,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工屯掖, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留玄柏,地道東北人。 一個月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓贴铜,卻偏偏與公主長得像粪摘,于是被迫代替她去往敵國和親瀑晒。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,627評論 2 350

推薦閱讀更多精彩內(nèi)容