淺談es6中的元編程
何為元編程?
「編寫(xiě)能改變語(yǔ)言語(yǔ)法特性或者運(yùn)行時(shí)特性的程序」。換言之,一種語(yǔ)言本來(lái)做不到的事情,通過(guò)你編程來(lái)修改它,使得它可以做到了,這就是元編程。
meta-programming元編程中的 元 的概念可以理解為 程序 本身?!痹幊棠茏屇銚碛锌梢詳U(kuò)展程序自身能力
舉個(gè)例子:
if (a == 1 && a == 2 && a == 3) { console.log("done"); }
怎樣才能讓這個(gè)條件滿足,輸出done。按照正常的邏輯是無(wú)法完成的,畢竟一個(gè)值不可能同時(shí)滿足等于1、2、3
這是就可以用到元編程來(lái)改變這個(gè)不可能
let a = { [Symbol.toPrimitive]: ((i) => () => ++i)(0) } if (a == 1 && a == 2 && a == 3) { console.log("done"); } // done
Symbol.toPrimitive在對(duì)象轉(zhuǎn)換為原始值的時(shí)候會(huì)被調(diào)用,初始值為1,調(diào)用一次+1,就可以滿足a == 1 && a == 2 && a == 3,同時(shí)Symbol.toPrimitive也可以接受一個(gè)參數(shù)hint,hint的取值為number、string、default
let obj = { [Symbol.toPrimitive](hint) { switch (hint) { case "number": return 123; case "string": return "str"; case "default": return "default"; } } } console.log(1-obj); // -122 console.log(1+obj); // 1default console.log(`${obj}`); // str
還有哪些元編程?
proxy
es5的Object.defineProperty()方法的es6升級(jí)版,用于自定義的對(duì)象的行為
let leon = { age: 30 } const validator = { get: function(target, key){ // 若沒(méi)這個(gè)屬性返回37 return key in target ? target[key] : 37; }, set(target,key,value){ if(typeof value!="number" || Number.isNaN(value)){ throw new Error("年齡得是數(shù)字"); } } } const proxy = new Proxy(leon,validator); console.log(proxy.name); // 37 proxy.age = "hi"; // Error: 年齡得是數(shù)字
reflect-metadata
你可以通過(guò)裝飾器來(lái)給類添加一些自定義的信息。然后通過(guò)反射將這些信息提取出來(lái)。當(dāng)然你也可以通過(guò)反射來(lái)添加這些信息
require("reflect-metadata") class C { // @Reflect.metadata(metadataKey, metadataValue) method() { } } Reflect.defineMetadata("name", "jix", C.prototype, "method"); let metadataValue = Reflect.getMetadata("name", C.prototype, "method"); console.log(metadataValue); // jix
應(yīng)用
拓展數(shù)組索引訪問(wèn)
負(fù)索引訪問(wèn),使array[-N] 與 array[array.length - N] 相同
let array = [1, 2, 3]; array = new Proxy(array, { get(target, prop, receiver) { if (prop < 0) { console.log(prop, 'prop') prop = +prop + target.length; } return Reflect.get(target, prop, receiver); } }); console.log(array[-1]); // 3 console.log(array[-2]); // 2
數(shù)據(jù)劫持
let handlers = Symbol('handlers'); function makeObservable(target) { // 初始化存儲(chǔ) handler 的數(shù)組 target[handlers] = []; // 存儲(chǔ) handler 函數(shù)到數(shù)組中以便于未來(lái)調(diào)用 target.observe = function(handler) { this[handlers].push(handler); }; // 創(chuàng)建代理以處理更改 return new Proxy(target, { set(target, property, value, receiver) { // 轉(zhuǎn)發(fā)寫(xiě)入操作到目標(biāo)對(duì)象 let success = Reflect.set(...arguments); // 如果設(shè)置屬性的時(shí)候沒(méi)有報(bào)錯(cuò) if (success) { // 調(diào)用所有 handler target[handlers].forEach(handler => handler(property, value)); } return success; } }); } let user = {}; user = makeObservable(user); user.observe((key, value) => { console.log(`SET ${key}=${value}`); }); user.name = "John"; // SET name=John
到此這篇關(guān)于淺談es6中的元編程的文章就介紹到這了,更多相關(guān)es6 元編程內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
調(diào)整小數(shù)的格式保留小數(shù)點(diǎn)后兩位
調(diào)整小數(shù)的格式,如保留小數(shù)點(diǎn)后兩位等等在開(kāi)發(fā)過(guò)程中經(jīng)常會(huì)遇到,下面本文搜集了一些不錯(cuò)的實(shí)現(xiàn)方法與分享2014-05-05uniapp地圖組件(map)使用與遇到的一些問(wèn)題總結(jié)
uniapp是用vue.js開(kāi)發(fā)所有前端應(yīng)用的框架,開(kāi)發(fā)人員只需要編寫(xiě)一套代碼就可以發(fā)布到安卓、iOS、H5和小程序、快應(yīng)用等平臺(tái),下面這篇文章主要給大家介紹了關(guān)于uniapp地圖組件(map)使用與遇到的一些問(wèn)題,需要的朋友可以參考下2022-07-07JS實(shí)現(xiàn)頁(yè)面超時(shí)后自動(dòng)跳轉(zhuǎn)到登陸頁(yè)面
這篇文章主要介紹了JS實(shí)現(xiàn)頁(yè)面超時(shí)后自動(dòng)跳轉(zhuǎn)到登陸頁(yè)面,需要的朋友可以參考下2015-01-01JavaScript實(shí)現(xiàn)預(yù)覽本地上傳圖片功能完整示例
這篇文章主要介紹了JavaScript實(shí)現(xiàn)預(yù)覽本地上傳圖片功能,結(jié)合完整實(shí)例形式分析了javascript圖片預(yù)覽相關(guān)的格式正則驗(yàn)證、瀏覽器判斷、頁(yè)面元素屬性動(dòng)態(tài)操作相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-03-03Express實(shí)現(xiàn)前端后端通信上傳圖片之存儲(chǔ)數(shù)據(jù)庫(kù)(mysql)傻瓜式教程(二)
這篇文章主要介紹了Express實(shí)現(xiàn)前端后端通信上傳圖片之存儲(chǔ)數(shù)據(jù)庫(kù)(mysql)傻瓜教程(二)的相關(guān)資料,需要的朋友可以參考下2015-12-12利用JavaScript實(shí)現(xiàn)檢測(cè)用戶是否在線功能
這篇文章主要為大家詳細(xì)介紹了如何利用JavaScript實(shí)現(xiàn)檢測(cè)用戶是否在線功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-12-12webpack 插件html-webpack-plugin的具體使用
本篇文章主要介紹了webpack 插件html-webpack-plugin的具體使用,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-04-04