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

資訊專欄INFORMATION COLUMN

用JS創(chuàng)造一個全新的編程語言

muzhuyu / 3169人閱讀

摘要:編程語言代碼改自編程語言遵循如下規(guī)范語言只有表達(dá)式,沒有語句,如加法,連續(xù)相加如多個表達(dá)式用包裹,如數(shù)據(jù)類型支持?jǐn)?shù)字布爾字符串代碼實現(xiàn)將源碼字符串成。一個合法的語言必然是一個表達(dá)式,所以解析的開始可以是函數(shù),如解析源碼。

egg編程語言

代碼改自Eloquent Javascript: Chapter 12

egg編程語言遵循如下規(guī)范

語言只有表達(dá)式,沒有語句,如加法:+(1, 7),連續(xù)相加如+(1, +(2,3))

多個表達(dá)式用do包裹,如do(expr1, expr2, expr3, ..., exprN)

數(shù)據(jù)類型支持?jǐn)?shù)字 布爾 字符串

代碼實現(xiàn): parse

將源碼字符串parse成ast。

do(foo(), "done")
{
    "type": "apply",
    "operator": { "type": "word", "name": "do" },
    "args": [
        {
            "type": "apply",
            "operator": { "type": "word", "name": "foo" },
            "args": []
        },
        { "type": "value", "value": "done" }
    ]
}

一個合法的egg語言必然是一個表達(dá)式,所以解析的開始可以是parseExpression函數(shù),如解析egg源碼do(expr)。解析的源碼也可以是僅僅是普通的字符串"abc",數(shù)字3.14或者僅僅是變量Identifer,這就需要判斷是不是函數(shù)調(diào)用,即解析的表達(dá)式后面有沒有"("。
所以我們先構(gòu)建一個parse函數(shù)作為代碼的包裹:

// 全局program變量
var program = "";
function parse(_program) {
    program = _program // 更新全局變量
    var result = parseExpression();
    // 源碼只能有一個表達(dá)式,如果解析完了還有剩余字符拋出異常。
    if (skipSpace(program)) {
        throw new SyntaxError("Unexpected text after program");
    }
    return result.expr;
}

// 過濾空白字符
function skipSpace(string) {
    var first = string.search(/S/);
    if(first === -1) 
        return "";
    return string.slice(first);
}
// 解析一個表達(dá)式 do(expr)
function parseExpression() {
    program = skipSpace(program);
    var expr;
    if (/^[""]/.test(program)) {
        // 表達(dá)式是字符串
        expr = { type: "value", value: parseString() };
    } else if (/^[d.]/.test(program)) {
        // 表達(dá)式是數(shù)字
        expr = { type: "value", value: parseNumber() };
    } else if (/^[^s(),"]/.test(program)) {
        表達(dá)式是字母(變量名標(biāo)識及true、false等)
        expr = { type: "word", name: parseWord() };
    } else throw new SyntaxError("Unexpected syntax: " + program);
        /** 解析完單個表達(dá)式do(),截止目前結(jié)果如 {type: "word", name: "do"},
         *  還需要判斷是否為函數(shù)調(diào)用
         */
    return parseApply(expr);
}

function parseApply(expr) {
    program = skipSpace(program);
    // 如果變量名后不是"(",就不是函數(shù)調(diào)用,直接返回parseExpression返回的解析后的表達(dá)式
    if (program[0] != ")")
        return {
            expr: expr
        }
    // 如果是函數(shù)調(diào)用
    program = skipSpace(program.slice(1));
    // expr需要被apply調(diào)用形式的expr包裹,原expr為operator
    expr = {type: "apply", operator: expr, args: []};
    while (program[0] != ")") {
        // 解析參數(shù)
        var arg = parseExpression();
        expr.args.push(arg.expr);
        program = skipSpace(program);
        if (program[0] == ",")
            program = program.slice(1);
        else if (program[0] != ")")
            throw new SyntaxError("Expected , or )");        
    }
    program = program.substr(1);
    return parseApply(expr); // 遞歸解析 foo()()形式的表達(dá)式;
}

// parseExpression 需要解析三種類型的表達(dá)式
/** string */

function parseString() {
    var startSign = program[0],
        value = "",
        at = 0,
        ch;
    while ((ch = program[++at])) {
        if (ch == "") {
            value += program[++at];
            continue;
        }
        if (ch == startSign) {
            program = program.substr(++at);
            return value;
        }
        value += ch;
    }
    throw new SyntaxError("Parse String Error");
}

/** number */
function parseNumber() {
    var at = 0,
        ch,
        value = "";
    while ((ch = program[at]) && /[d.]/.test(ch)) {
        value += ch;
        at++;
    }
    program = program.substr(at);
    value = Number(value);
    if (isNaN(value)) throw new SyntaxError("Parse Number Error");
    return value;
}
/** word */

function parseWord() {
    var at = 0,
        ch,
        value = "";
    while ((ch = program[at]) && /[^s(),"]/.test(ch)) {
        value += ch;
        at++;
    }
    program = program.substr(at);
    return value;
}
evaluate
function evalute(expr, env) {
    switch(expr.type) {
        case "value":
            return expr.value;
        case "word":
            if (expr.name in env)
                return env[expr.name];
            else
                throw new ReferenceError("undefined variable: "
                + expr.name);
        case "apply":
            if (
                expr.operator.type == "word" &&
                expr.operator.name in specialForms
            )
                return specialForms[expr.operator.name](expr.args, env);
            var op = evalute(expr.operator, env);
            if (typeof op != "function")
                throw new TypeError("Applying a non-function");
            return op.apply(null, expr.args.map(function(arg) {
                return evalute(arg, env);
            });
    }
}

/** 原生函數(shù) */
var specialForms = Object.create(null);

specialForms["if"] = function(args, env) {
    if (args.length != 3) throw new SyntaxError("Bad number of args to if");
    if (evaluate(args[0], env) !== false) return evaluate(args[1], env);
    else return evaluate(args[2], env);
};

specialForms["while"] = function(args, env) {
    if (args.length != 2)
        throw new SyntaxError("Bad number of args to while");
    while (evalute(args[0], env) !== false)
        evalute(args[1], env);
    return false;
}

specialForms["do"] = function(args, env) {
    var value = false;
    args.forEach(function(arg) {
        value = evaluate(arg, env);
    });
    return value;
};

specialForms["define"] = function(args, env) {
    if (args.length != 2 || args[0].type != "word")
        throw new SyntaxError("Bad use of define");
    var value = evaluate(args[1], env);
    env[args[0].name] = value;
    return value;
}

// 自定義函數(shù)擴展
specialForms["fun"] = function(args, env) {
    if (!args.length) throw new SyntaxError("Function need a body");
    function name(expr) {
        if (expr.type != "word")
            throw new SyntaxError("Arg names must be words");
        return expr.name;
    }
    var argNames = args.slice(0, args.length - 1).map(name);
    var body = args[args.length - 1];
    return function() {
        if (arguments.length != argNames.length)
            throw new TypeError("Wrong number of arguments");
        var localEnv = Object.create(env);
        for (var i = 0; i < arguments.length; i++)
            localEnv[argNames[i]] = arguments[i];
        return evaluate(body, localEnv);
    }
}

// 擴展原生操作符
var topEnv = Object.create(null);
topEnv["true"] = true;
topEnv["false"] = false;

["+", "-", "*", "/", "==", "<", ">"].forEach(function(op) {
    topEnv[op] = new Function("a", "b", "return a" + op + "b");
});

topEnv["print"] = function(value) {
    console.log(value);
    return value;
};

function run() {
    var env = Object.create(topEnv);
    var program = Array.prototype.slice.call(arguments, 0).join("
");
    return evaluate(parse(program), env);
}
斐波那契數(shù)列
#! /usr/bin/env node

const run = require("./egg").run;
run(
    `
    do(
        define(foo, fun(
            n,
            if(<(n, 3), 1, +(foo(-(n, 1)), foo(-(n, 2))))
        )),
        print(foo(7))
    )
`);

完整項目見 Github: https://github.com/lcfme/egg-...

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

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

相關(guān)文章

  • 全新單頁JavaScript框架Amaple

    摘要:但目前來看,單頁應(yīng)用在技術(shù)實現(xiàn)和體驗上還有更大的發(fā)展空間,而這就是所要推進的。模塊化頁面模塊化單頁應(yīng)用的特點之一是將頁面劃分為多個模塊,跳轉(zhuǎn)時更新模塊的內(nèi)容。與其他單頁庫相比,它們的職責(zé)更清晰,也易于理解。 showImg(https://segmentfault.com/img/bV2wO3?w=792&h=303);單頁Web應(yīng)用作為新一代Web模式,為用戶提供了更流暢的體驗滿足感...

    djfml 評論0 收藏0
  • 全新單頁JavaScript框架Amaple

    摘要:但目前來看,單頁應(yīng)用在技術(shù)實現(xiàn)和體驗上還有更大的發(fā)展空間,而這就是所要推進的。模塊化頁面模塊化單頁應(yīng)用的特點之一是將頁面劃分為多個模塊,跳轉(zhuǎn)時更新模塊的內(nèi)容。與其他單頁庫相比,它們的職責(zé)更清晰,也易于理解。 showImg(https://segmentfault.com/img/bV2wO3?w=792&h=303);單頁Web應(yīng)用作為新一代Web模式,為用戶提供了更流暢的體驗滿足感...

    yanwei 評論0 收藏0
  • 全新單頁JavaScript框架Amaple

    摘要:但目前來看,單頁應(yīng)用在技術(shù)實現(xiàn)和體驗上還有更大的發(fā)展空間,而這就是所要推進的。模塊化頁面模塊化單頁應(yīng)用的特點之一是將頁面劃分為多個模塊,跳轉(zhuǎn)時更新模塊的內(nèi)容。與其他單頁庫相比,它們的職責(zé)更清晰,也易于理解。 showImg(https://segmentfault.com/img/bV2wO3?w=792&h=303);單頁Web應(yīng)用作為新一代Web模式,為用戶提供了更流暢的體驗滿足感...

    wemallshop 評論0 收藏0
  • FCC 成都社區(qū)·前端周刊 第 10 期

    摘要:正式發(fā)布在過去的一周,正式發(fā)布,帶來大量改進和修復(fù)。這是自開展以來的第七個主要版本,并將在年月成為下一個分支。以后,如果使用具有已知安全問題的代碼,的用戶會收到警告通知。將自動檢查針對數(shù)據(jù)庫的安裝請求,并在代碼包含漏洞時發(fā)出警告通知。 1. Node.js 10 正式發(fā)布 在過去的一周,Node.js 10.0.0 正式發(fā)布,帶來大量改進和修復(fù)。這是自 Node.js Foundati...

    BigNerdCoding 評論0 收藏0

發(fā)表評論

0條評論

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