摘要:中,便是兩個可以用來進(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
摘要:在這里講到的很多也許只和程序?qū)τ诠ぷ鳈C(jī)制的操作有關(guān),但是作為初探也許也就足夠了。一般情況下還有空字符串都會被判斷成。面向特征編程面向特征編程的全稱是。 引子 元編程會有如下的定義: 一種計算機(jī)程序的編寫方式,它可以將其它程序(或者其本身)作為數(shù)據(jù)進(jìn)行編寫和操作,或者在編譯時做一部分工作,在運行的時候做另外一部分工作。 在這里講到的很多也許只和程序?qū)τ诠ぷ鳈C(jī)制的操作有關(guān),但是作為初...
摘要:事實上,實現(xiàn)元編程有多種方式,從語言本身來講,可以分為兩類增強(qiáng)型與新的語法實現(xiàn),前者的代表是反射,后者的代表為。在第二部分,我們嘗試在語言基礎(chǔ)上增加原生的元編程能力并介紹了該思路的實現(xiàn)框架。 語言的自由度 自由度這個概念在不同領(lǐng)域有不同的定義,我們借鑒數(shù)學(xué)中構(gòu)成一個空間的維數(shù)來表達(dá)其自由度的做法,在此指的是:解決同一個問題彼此不相關(guān)的設(shè)計方法學(xué)數(shù)量。 例如,解決一個比如商品打折的問題,...
摘要:事實上,實現(xiàn)元編程有多種方式,從語言本身來講,可以分為兩類增強(qiáng)型與新的語法實現(xiàn),前者的代表是反射,后者的代表為。在第二部分,我們嘗試在語言基礎(chǔ)上增加原生的元編程能力并介紹了該思路的實現(xiàn)框架。 語言的自由度 自由度這個概念在不同領(lǐng)域有不同的定義,我們借鑒數(shù)學(xué)中構(gòu)成一個空間的維數(shù)來表達(dá)其自由度的做法,在此指的是:解決同一個問題彼此不相關(guān)的設(shè)計方法學(xué)數(shù)量。 例如,解決一個比如商品打折的問題,...
摘要:邏輯運算一般語言中,邏輯運算與布爾元算是等義的,其運算元與目標(biāo)類型都是布爾值。除此之外,還有以下的兩條特性運算符會將運算元理解為布爾值,以進(jìn)行布爾運算。運算過程是支持布爾短路的。 邏輯運算 一般語言中,邏輯運算與布爾元算是等義的,其運算元與目標(biāo)類型都是布爾值。JavaScript當(dāng)然支持這種純布爾運算,不但如此,JavaScript還包括另外一種邏輯運算,它的表達(dá)式結(jié)果是不確定的。 ...
閱讀 2896·2021-09-28 09:36
閱讀 3655·2021-09-27 13:59
閱讀 2500·2021-08-31 09:44
閱讀 2288·2019-08-30 15:54
閱讀 2361·2019-08-30 15:44
閱讀 1196·2019-08-30 13:45
閱讀 1232·2019-08-29 18:38
閱讀 1222·2019-08-29 18:37