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

資訊專欄INFORMATION COLUMN

Javascript元編程(一)

mo0n1andin / 1478人閱讀

摘要:中,便是兩個可以用來進(jìn)行元編程的特性。這也就是元編程的優(yōu)點之一,程序可以根據(jù)傳入?yún)?shù)對象的不同,動態(tài)地生成對應(yīng)的程序,從而減少大量冗余的代碼。

首發(fā)于知乎專欄:http://zhuanlan.zhihu.com/starkwang

這幾天把一年多前買的《松本行弘的程序世界》重新看了看,很多當(dāng)時不能理解的東西現(xiàn)在再去看真是茅塞頓開呀,看到元編程那一段真是把我震撼到了,后來發(fā)現(xiàn) Javascript 里其實也是有一些支持元編程的特性的,今天就用一個 DEMO 示范一下吧。

什么元編程

“元編程”這個名字看起來高端大氣上檔次,它的含義也是相當(dāng)高端:“寫一段自動寫程序的程序”,不要誤會,我們做的可不是人工智能。

言簡意賅地說,元編程就是將代碼視作數(shù)據(jù),直接用字符串 or AST or 其他任何形式去操縱代碼,以此獲得一些維護(hù)性、效率上的好處。

Javascript 中,eval、new Function()便是兩個可以用來進(jìn)行元編程的特性。

原始示例

現(xiàn)在我們有一堆用戶的數(shù)據(jù),具體字段有name,sex,age,address等等,通過類似 /get_name?id=123456 來拉取數(shù)據(jù)

那么我們很容易寫出這樣的代碼:

class User {
    constructor(userID) {
        this.id = userID;
    }

    get_name() {
        return $.ajax(`/get_name?id=${this.id}`);
    }

    get_sex() {
        return $.ajax(`/get_sex?id=${this.id}`);
    }

    //下面是get_age、get_address......
}

這段代碼的問題在哪呢?

首先,用戶數(shù)據(jù)有多少個字段,我們就要定義多少個 get_something 方法,更可怕的是這些方法里邏輯都是重復(fù)的,都是一個簡單的 ajax。

進(jìn)階(一)

我們可以把拉取數(shù)據(jù)的邏輯封裝到 __fetchData 里:

class User {
    constructor(userID) {
        this.id = userID;
    }
    
    __fetchData(key) {
        //這是一個private方法,直接調(diào)用類似__fetchData("age")是不被允許的
        return $.ajax(`/get_${key}?id=${this.id}`)
    }

    get_name() {
        return this.__fetchData("name");
    }

    get_sex() {
        return this.__fetchData("sex");
    }

    //下面是get_age、get_address......
}

然后,冗余的問題可以通過registerProperties來解決:

class User {
    constructor(userID) {
        this.id = userID;
        this.registerProperties(["name", "age", "sex", "address"]);
    }

    registerProperties(keyArray) {
        keyArray.forEach(key => {
            this[`get_${key}`] = () => this.__fetchData(key);
        })
    }

    __fetchData(key) {
        //這是一個private方法,直接調(diào)用類似__fetchData("age")是不被允許的
        return $.ajax(`/get_${key}?id=${this.id}`)
    }
}
進(jìn)階(三)

到目前為止我們都沒有涉及到任何元編程的概念,下面我們加上更高的需求:

在拉去數(shù)據(jù)之后,我們要對部分?jǐn)?shù)據(jù)進(jìn)行一定的處理,比如對 name 我們要去掉首尾的空格,對 age 我們要加上一個 字。具體的處理方法定義在 __handle_something 里面。

這里我們便可以通過 new Function() 來動態(tài)生成函數(shù),元編程開始顯現(xiàn)威力:

class User {
    constructor(userID) {
        this.id = userID;
        this.registerProperties(["name", "age", "sex", "address"]);
    }

    registerProperties(keyArray) {
        keyArray.forEach(key => {
            //注意這里的fnBody內(nèi)部依然采用ES5的寫法,因為babel目前不會編譯函數(shù)字符串。
            var fnBody = `return this.__fetchData("/get_${key}?id=${this.id}")
                    .then(function(data){
                        return this.__handle_${key}?_this.handle_${key}(data):data;
                    })`;
            this[`get_${key}`] = new Function(fnBody);
        })
    }

    __handle_name(name) {
        //do somthing with name...
        return name;
    }

    __handle_age(age) {
        //do somthing with age...
        return age;
    }

    __fetchData(key) {
        //這是一個private方法,直接調(diào)用類似__fetchData("age")是不被允許的
        return $.ajax(`/get_${key}?id=${this.id}`)
    }
}
進(jìn)階(四)

下面我們讓需求更加{{BANNED}}一點:

數(shù)據(jù)并非通過 ajax 直接拉取,而是通過一個別人封裝好的 UserDataBase 里的方法來拉取;

數(shù)據(jù)的字段并非只有name,sex,age,address四個,而是要根據(jù) UserDataBase 里給你的方法決定。給你1000個get不同字段的方法,User類里也要有對應(yīng)的1000個方法。

class UserDataBase {
    constructor() {}
    get_name(id) {}
    get_age(id) {}
    get_address(id) {}
    get_sex(id) {}
    get_anything_else1(id) {}
    get_anything_else2(id) {}
    get_anything_else3(id) {}
    get_anything_else4(id) {}
    //......
}

這里我們就需要用到 JS 的反射機(jī)制來讀取所有拉取字段的方法,然后通過元編程的方式來動態(tài)生成對應(yīng)的方法。

class User {
    constructor(userID, dataBase) {
        this.id = userID;
        this.__dataBase = dataBase;
        for (var method in dataBase) {
            //對每一個方法
            this.registerMethod(method);
        }
    }

    registerMethod(methodName) {
        //這里除去了前置的"get_"
        var propertyName = methodName.slice(4);
        
        //注意這里拉取數(shù)據(jù)的方法改為使用dataBase
        var fnBody = `return this.__dataBase.${methodName}()
                    .then(function(data){
                        return this.__handle_${propertyName}?_this.handle_${propertyName}(data):data;
                    })`;
        this[`get_${propertyName}`] = new Function(fnBody);
    }

    __handle_name(name) {
        //do somthing with name...
        return name;
    }

    __handle_age(age) {
        //do somthing with age...
        return age;
    }
}
var userDataBase = new UserDataBase();
var user = new User("123", userDataBase);

這樣即使用戶數(shù)據(jù)有一萬種不同的屬性字段,只要保證 UserDataBase 中良好地定義了對應(yīng)的拉取方法,我們的 User 就能自動生成對應(yīng)的方法。

這也就是元編程的優(yōu)點之一,程序可以根據(jù)傳入?yún)?shù)/對象的不同,動態(tài)地生成對應(yīng)的程序,從而減少大量冗余的代碼。

進(jìn)階(五)

現(xiàn)在程序里還有點小瑕疵:

//用戶數(shù)據(jù)中不存在www字段,若這樣執(zhí)行會報錯:
user.get_www(); //user.get_www is not a function

現(xiàn)在我們要保證像上面那樣執(zhí)行任意的 user.get_xxxx() ,程序不會報錯,而是返回 false

//用戶數(shù)據(jù)中不存在www字段:
user.get_www(); // => false

Javascript 里缺少了 Ruby 中 method_missing 這樣黑科技的內(nèi)核方法,但是我們可以通過 ES6 的 Proxy 特性來模擬:

function createUser(id, userDataBase) {
    return new Proxy(new User(id, userDataBase), {
        get: (target, property) => (typeof(target[property]) === "function" ? target[property] : () => false)
    })
}

var userDataBase = new UserDataBase();
var user = createUser("123", userDataBase);

user.get_name() => // fetch name data
user.get_wwwwww() // => false
總結(jié)

其實這里的 DEMO 只是元編程的一個小應(yīng)用,下一篇文章里我們會通過元編程實現(xiàn)一個簡單的表單驗證 DSL :

//類似
form.name["is not empty"]["length is between",1,20] // => true or false
參考

來來來,咱么元編程入個門

元編程之javascript

JavaScript 元編程之ES6 Proxy

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

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

相關(guān)文章

  • Javascript 編程初探 [1]

    摘要:在這里講到的很多也許只和程序?qū)τ诠ぷ鳈C(jī)制的操作有關(guān),但是作為初探也許也就足夠了。一般情況下還有空字符串都會被判斷成。面向特征編程面向特征編程的全稱是。 引子 元編程會有如下的定義: 一種計算機(jī)程序的編寫方式,它可以將其它程序(或者其本身)作為數(shù)據(jù)進(jìn)行編寫和操作,或者在編譯時做一部分工作,在運行的時候做另外一部分工作。 在這里講到的很多也許只和程序?qū)τ诠ぷ鳈C(jī)制的操作有關(guān),但是作為初...

    mcterry 評論0 收藏0
  • Javascript編程之Annotation

    摘要:事實上,實現(xiàn)元編程有多種方式,從語言本身來講,可以分為兩類增強(qiáng)型與新的語法實現(xiàn),前者的代表是反射,后者的代表為。在第二部分,我們嘗試在語言基礎(chǔ)上增加原生的元編程能力并介紹了該思路的實現(xiàn)框架。 語言的自由度 自由度這個概念在不同領(lǐng)域有不同的定義,我們借鑒數(shù)學(xué)中構(gòu)成一個空間的維數(shù)來表達(dá)其自由度的做法,在此指的是:解決同一個問題彼此不相關(guān)的設(shè)計方法學(xué)數(shù)量。 例如,解決一個比如商品打折的問題,...

    SegmentFault 評論0 收藏0
  • Javascript編程之Annotation

    摘要:事實上,實現(xiàn)元編程有多種方式,從語言本身來講,可以分為兩類增強(qiáng)型與新的語法實現(xiàn),前者的代表是反射,后者的代表為。在第二部分,我們嘗試在語言基礎(chǔ)上增加原生的元編程能力并介紹了該思路的實現(xiàn)框架。 語言的自由度 自由度這個概念在不同領(lǐng)域有不同的定義,我們借鑒數(shù)學(xué)中構(gòu)成一個空間的維數(shù)來表達(dá)其自由度的做法,在此指的是:解決同一個問題彼此不相關(guān)的設(shè)計方法學(xué)數(shù)量。 例如,解決一個比如商品打折的問題,...

    lifesimple 評論0 收藏0
  • javascript語言精髓與編程實踐摘要

    摘要:邏輯運算一般語言中,邏輯運算與布爾元算是等義的,其運算元與目標(biāo)類型都是布爾值。除此之外,還有以下的兩條特性運算符會將運算元理解為布爾值,以進(jìn)行布爾運算。運算過程是支持布爾短路的。 邏輯運算 一般語言中,邏輯運算與布爾元算是等義的,其運算元與目標(biāo)類型都是布爾值。JavaScript當(dāng)然支持這種純布爾運算,不但如此,JavaScript還包括另外一種邏輯運算,它的表達(dá)式結(jié)果是不確定的。 ...

    shiina 評論0 收藏0

發(fā)表評論

0條評論

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