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

資訊專欄INFORMATION COLUMN

基于JavaScript的簡單解釋器實(shí)現(xiàn)(二)——函數(shù)解析與執(zhí)行

greatwhole / 3261人閱讀

摘要:函數(shù)執(zhí)行函數(shù)執(zhí)行使用后續(xù)遍歷的方式來遍歷語法樹。對于每一個子節(jié)點(diǎn),若其為函數(shù)則遞歸調(diào)用執(zhí)行函數(shù)。如果當(dāng)前方法是運(yùn)算符方法,則調(diào)用該運(yùn)算符的執(zhí)行函數(shù),并返回結(jié)果如果當(dāng)前方法是函數(shù),則解析所有形參的值后生產(chǎn)函數(shù)作用域,并以改作用域執(zhí)行當(dāng)前函數(shù)。

前言

昨晚奮斗了一下,終于把這題了解了。今天完善了一下代碼,把剩下的部分放上來。目前剩下的有兩個主要模塊即函數(shù)解析與函數(shù)執(zhí)行,以及兩個小模塊即運(yùn)算符執(zhí)行和變量解析。
題目地址:http://www.codewars.com/kata/52ffcfa4aff455b3c2000750/train/javascript
github地址:https://github.com/woodensail/SimpleInteractiveInterpreter
前文地址:http://segmentfault.com/a/1190000004044789
本文地址:http://segmentfault.com/a/1190000004047915

函數(shù)解析
var index = tokens.indexOf("=>"), paramObj = {}, params = [], fnName = tokens[1];

初始化參數(shù),paramObj用于統(tǒng)計(jì)函數(shù)體中用到的參數(shù),params為形參列表,index為函數(shù)運(yùn)算符的位置,fnName為函數(shù)名

if (this.vars[fnName] !== void 0) {
    throw "name conflicting"
}

如果全局變量中存在該名稱的變量,則拋出異常

for (var i = 2; i < index; i++) {
    if (paramObj[tokens[i]]) {
        throw "param conflicting"
    }
    paramObj[tokens[i]] = 1;
    params.push(tokens[i]);
}

統(tǒng)計(jì)形參,如果同名的形參則拋出異常。

var result = this.expressionParser(tokens.slice(index + 1));
var syntaxTree = result[0], varList = result[1];
varList.forEach(function (v) {
    if (!paramObj[v]) {
        throw "nonexistent param"
    }
});
this.functions[fnName] = {params: params, syntaxTree: syntaxTree}

調(diào)用表達(dá)式解析器解析函數(shù)體部分。檢查函數(shù)體中用到的參數(shù),如果存在形參列表中不存在的參數(shù)則拋出異常。
最后將該函數(shù)存入函數(shù)表。

變量解析
Interpreter.prototype.extractValue = function (key, scope) {
    scope = scope || {};
    var value = scope[key];
    if (value === void 0) {
        value = this.vars[key];
    }
    if (value === void 0) {
        value = key;
    }
    if ("number" === typeof value) {
        return value;
    }
    throw "nonexistent var";
};

按照就優(yōu)先級分別嘗試提取作用域中的變量和全局變量以及key自身。提取完畢后若value不為number則所請求的值不存在。

運(yùn)算符實(shí)現(xiàn)
Interpreter.prototype.add = function (x, y, scope) {
    return this.extractValue(x, scope) + this.extractValue(y, scope);
};
Interpreter.prototype.sub = function (x, y, scope) {
    return this.extractValue(x, scope) - this.extractValue(y, scope);
};
Interpreter.prototype.mul = function (x, y, scope) {
    return this.extractValue(x, scope) * this.extractValue(y, scope);
};
Interpreter.prototype.div = function (x, y, scope) {
    return this.extractValue(x, scope) / this.extractValue(y, scope);
};
Interpreter.prototype.mod = function (x, y, scope) {
    return this.extractValue(x, scope) % this.extractValue(y, scope);
};
Interpreter.prototype.assign = function (x, y, scope) {
    var value = this.extractValue(y, scope);
    if (scope.x !== void 0) {
        return scope[x] = value;
    } else if ("number" === typeof x) {
        throw "assign to lValue"
    } else if (!this.functions[x]) {
        return this.vars[x] = value;
    }
    throw "name conflicting"
};

加減乘除模沒什么特殊的就是解析變量后運(yùn)算然后返回結(jié)果即可。
賦值語句需要對被賦值變量進(jìn)行判斷,如果當(dāng)前函數(shù)作用域中有該變量則賦值后返回,如果被賦值對象為數(shù)字,則拋出左值異常。如果函數(shù)表中不存在對應(yīng)函數(shù)則存入全局變量,否則拋出重名異常。

函數(shù)執(zhí)行

函數(shù)執(zhí)行使用后續(xù)遍歷的方式來遍歷語法樹。先依次計(jì)算每個參數(shù)的結(jié)果后,再用獲得的結(jié)果集執(zhí)行根節(jié)點(diǎn)。

Interpreter.prototype.exec = function (syntaxTree, scope) {
    scope = scope || {};
    ……
};

形參為語法樹和作用域。若未指定作用域則新建空作用域。

for (var i = 1; i < syntaxTree.length; i++) {
    if (syntaxTree[i] instanceof Array) {
        syntaxTree[i] = this.exec(syntaxTree[i], scope);
    }
}

對于每一個子節(jié)點(diǎn),若其為函數(shù)則遞歸調(diào)用執(zhí)行函數(shù)。這一步執(zhí)行完畢后當(dāng)前參數(shù)列表中應(yīng)該只存在變量或數(shù)字立即量。

if (this.native[name]) {
    params = syntaxTree.slice(1);
    params.push(scope);
    return this.native[name].apply(this, params);
}

如果當(dāng)前方法是運(yùn)算符方法,則調(diào)用該運(yùn)算符的執(zhí)行函數(shù),并返回結(jié)果

else if (this.functions[name]) {
    var fun = this.functions[name];
    params = {};

    fun.params.forEach(function (key, i) {
        var k = syntaxTree[i + 1];
        params[key] = _this.extractValue(k, scope);
    });

    return this.exec(fun.syntaxTree, params);
}

如果當(dāng)前方法是函數(shù),則解析所有形參的值后生產(chǎn)函數(shù)作用域,并以改作用域執(zhí)行當(dāng)前函數(shù)。

 else {
    return this.extractValue(syntaxTree, scope);
}

如果不是以上任一種,則當(dāng)前執(zhí)行的語句為數(shù)據(jù),直接提取后返回。

總結(jié)

一個基本的解釋器就算是完成了,有些沒有技術(shù)含量的銜接代碼我沒有貼上來,大家可以去git上看。這個解釋器再加上輸入輸出部分就可以構(gòu)成一個REPL了。順便,曬個AC圖。

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

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

相關(guān)文章

  • 【譯】小百行 JavaScript 打造 lambda 演算釋器

    摘要:在開始解析之前,先通過詞法分析器運(yùn)行源碼,這會將源碼打散成語法中全大寫的部分。我們基于每個規(guī)則的名稱的左側(cè)為其創(chuàng)建一個方法,再來看右側(cè)內(nèi)容如果是全大寫的單詞,說明它是一個終止符即一個,詞法分析器會用到它。 本文轉(zhuǎn)載自:眾成翻譯譯者:文藺鏈接:http://www.zcfy.cc/article/661原文:http://tadeuzagallo.com/blog/writing-a-l...

    KitorinZero 評論0 收藏0
  • JavaScript工作原理():V8引擎和5招高效代碼

    摘要:引擎可以用標(biāo)準(zhǔn)解釋器或即時(shí)編譯器來實(shí)現(xiàn),即時(shí)編譯器以某種形式將代碼編譯為字節(jié)碼。這里的主要區(qū)別在于不生成字節(jié)碼或任何中間代碼。請注意,不使用中間字節(jié)碼表示法,不需要解釋器。這允許在正常執(zhí)行期間非常短的暫停。 本系列的第一篇文章重點(diǎn)介紹了引擎,運(yùn)行時(shí)和調(diào)用棧的概述。第二篇文章將深入V8的JavaScript引擎的內(nèi)部。我們還會提供一些關(guān)于如何編寫更好的JavaScript代碼的技巧。 概...

    leone 評論0 收藏0
  • JavaScript語言特性以及重要版本

    摘要:通常一個完成的不僅僅包含了還包括了以及相關(guān)版本該版本在中使用?;谠秃瘮?shù)先行的語言使用基于原型的的繼承機(jī)制,函數(shù)是的第一等公民其他相關(guān)的語言特性編譯型語言把做好的源程序全部編譯成二進(jìn)制代碼的可運(yùn)行程序。 轉(zhuǎn)載請注明出處,創(chuàng)作不易,更多文章請戳 https://github.com/ZhengMaste... 前言:JavaScript誕生于1995年,它是一門腳本語言,起初的目...

    Yangder 評論0 收藏0
  • JavaScript 是如何工作解析、抽象語法樹(AST)+ 提升編譯速度5個技巧

    摘要:無論你使用的是解釋型語言還是編譯型語言,都有一個共同的部分將源代碼作為純文本解析為抽象語法樹的數(shù)據(jù)結(jié)構(gòu)。和抽象語法樹相對的是具體語法樹,通常稱作分析樹。這是引入字節(jié)碼緩存的原因。 這是專門探索 JavaScript 及其所構(gòu)建的組件的系列文章的第 14 篇。 想閱讀更多優(yōu)質(zhì)文章請猛戳GitHub博客,一年百來篇優(yōu)質(zhì)文章等著你! 如果你錯過了前面的章節(jié),可以在這里找到它們: JavaS...

    raoyi 評論0 收藏0
  • javascript引擎——V8

    摘要:類將源代碼解釋并構(gòu)建成抽象語法樹,使用類來創(chuàng)建它們,并使用類來分配內(nèi)存。類抽象語法樹的訪問者類,主要用來遍歷抽象語法樹。在該函數(shù)中,先使用類來生成抽象語法樹再使用類來生成本地代碼。 通過上一篇文章,我們知道了JavaScript引擎是執(zhí)行JavaScript代碼的程序或解釋器,了解了JavaScript引擎的基本工作原理。我們經(jīng)常聽說的JavaScript引擎就是V8引擎,這篇文章我們...

    luoyibu 評論0 收藏0

發(fā)表評論

0條評論

閱讀需要支付1元查看
<