亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > JS > 正文

新手快速入門JavaScript裝飾者模式與AOP

2024-05-06 16:52:41
字體:
來源:轉載
供稿:網友

什么是裝飾者模式

當我們拍了一張照片準備發朋友圈時,許多小伙伴會選擇給照片加上濾鏡。同一張照片、不同的濾鏡組合起來就會有不同的體驗。這里實際上就應用了裝飾者模式:是通過濾鏡裝飾了照片。在不改變對象(照片)的情況下動態的為其添加功能(濾鏡)。

需要注意的是:由于 JavaScript 語言動態的特性,我們很容易就能改變某個對象(JavaScript 中函數是一等公民)。但是我們要盡量避免直接改寫某個函數,這會導致代碼的可維護性、可擴展性變差,甚至會污染其他業務。

什么是 AOP

想必大家對"餐前洗手、飯后漱口"都不陌生。這句標語其實就是 AOP 在生活中的例子:吃飯這個動作相當于切點,我們可以在這個切點前、后插入其它如洗手等動作。

AOP(Aspect-Oriented Programming):面向切面編程,是對 OOP 的補充。利用AOP可以對業務邏輯的各個部分進行隔離,也可以隔離業務無關的功能比如日志上報、異常處理等,從而使得業務邏輯各部分之間的耦合度降低,提高業務無關的功能的復用性,也就提高了開發的效率。

在 JavaScript 中,我們可以通過裝飾者模式來實現 AOP,但是兩者并不是一個維度的概念。 AOP 是一種編程范式,而裝飾者是一種設計模式。

ES3 下裝飾者的實現

了解了裝飾者模式和 AOP 的概念之后,我們寫一段能夠兼容 ES3 的代碼來實現裝飾者模式:

// 原函數var takePhoto =function(){console.log('拍照片');}// 定義 aop 函數var after=function( fn, afterfn ){ return function(){let res = fn.apply( this, arguments ); afterfn.apply( this, arguments );return res;}}// 裝飾函數var addFilter=function(){console.log('加濾鏡');}// 用裝飾函數裝飾原函數takePhoto=after(takePhoto,addFilter);takePhoto();

這樣我們就實現了抽離拍照與濾鏡邏輯,如果以后需要自動上傳功能,也可以通過aop函數after來添加。

ES5 下裝飾者的實現

在 ES5 中引入了Object.defineProperty,我們可以更方便的給對象添加屬性:

let takePhoto = function () {console.log('拍照片');}// 給 takePhoto 添加屬性 afterObject.defineProperty(takePhoto, 'after', {writable: true,value: function () {console.log('加濾鏡');},});// 給 takePhoto 添加屬性 beforeObject.defineProperty(takePhoto, 'before', {writable: true,value: function () {console.log('打開相機');},});// 包裝方法let aop = function (fn) {return function () {fn.before()fn()fn.after()}}takePhoto = aop(takePhoto)takePhoto()

基于原型鏈和類的裝飾者實現

我們知道,在 JavaScript 中,函數也好,類也好都有著自己的原型,通過原型鏈我們也能夠很方便的動態擴展,以下是基于原型鏈的寫法:

class Test {takePhoto() {console.log('拍照');}}// after AOPfunction after(target, action, fn) {let old = target.prototype[action];if (old) {target.prototype[action] = function () {let self = this;fn.bind(self);fn(handle);}}}// 用 AOP 函數修飾原函數after(Test, 'takePhoto', () => {console.log('添加濾鏡');});let t = new Test();t.takePhoto();

使用 ES7 修飾器實現裝飾者

在 ES7 中引入了@decorator 修飾器的提案,參考阮一峰的文章。修飾器是一個函數,用來修改類的行為。目前Babel轉碼器已經支持。注意修飾器只能裝飾類或者類屬性、方法。三者的具體區別請參考 MDN Object.defineProperty ;而 TypeScript 的實現又有所不同:TypeScript Decorator。

接下來我們通過修飾器來實現對方法的裝飾:

function after(target, key, desc) {const { value } = desc;desc.value = function (...args) {let res = value.apply(this, args);console.log('加濾鏡')return res;}return desc;}class Test{@aftertakePhoto(){console.log('拍照')}}let t = new Test()t.takePhoto()

可以看到,使用修飾器的代碼非常簡潔明了。

場景:性能上報

裝飾者模式可以應用在很多場景,典型的場景是記錄某異步請求請求耗時的性能數據并上報:

function report(target, key, desc) {const { value } = desc;desc.value = async function (...args) {let start = Date.now();let res = await value.apply(this, args);let millis = Date.now()-start;// 上報代碼return res;}return desc;}class Test{@reportgetData(url){// fetch 代碼}}let t = new Test()t.getData()

這樣使用@report修飾后的代碼就會上報請求所消耗的時間。擴展或者修改report函數不會影響業務代碼,反之亦然。

場景:異常處理

我們可以對原有代碼進行簡單的異常處理,而無需侵入式的修改:

function handleError(target, key, desc) {const { value } = desc;desc.value = async function (...args) {let res;try{res = await value.apply(this, args);}catch(err){// 異常處理logger.error(err)}return res;}return desc;}class Test{@handleErrorgetData(url){// fetch 代碼}}let t = new Test()t.getData()

通過以上兩個示例我們可以看到,修飾器的定義很簡單,功能卻非常強大。

小結

我們一步一步通過高階函數、原型鏈、Object.defineProperty和@Decorator分別實現了裝飾者模式。接下來在回顧一下:

  • 裝飾者模式非常適合給業務代碼附加非業務相關功能(如日志上報),就如同給照片加濾鏡;
  • 裝飾者模式非常適合無痛擴展別人的代碼(你經常需要接手別人的項目吧)

有些朋友可能會覺得裝飾者模式和 vue 的 mixin 機制很像,其實他們都是“開放-封閉原則”和“單一職責原則”的體現。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,


注:相關教程知識閱讀請移步到JavaScript/Ajax教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
精品丝袜一区二区三区| 在线性视频日韩欧美| 国产成人精品综合久久久| 亚洲老板91色精品久久| 久久久久久久成人| 日韩在线欧美在线| 亚洲精品中文字| 国产在线一区二区三区| 麻豆精品精华液| 日韩视频一区在线| 欧美精品久久久久久久免费观看| 国产精品亚洲网站| www亚洲精品| 久久久久免费精品国产| 日韩av电影在线播放| 欧美理论电影在线观看| 欧美激情精品久久久久久黑人| 欧美亚洲伦理www| 亚洲区bt下载| 精品国产一区二区三区久久久| 久久久久一本一区二区青青蜜月| 最新国产精品拍自在线播放| 欧美国产日韩精品| 欧美成人一区二区三区电影| 91精品国产色综合| 国产精品一区电影| 91黑丝在线观看| 亚洲精品美女久久久久| 亚洲片在线观看| 日韩av在线免费| 欧美在线免费观看| 亚洲影院色在线观看免费| 九九热精品视频| 亚洲欧美在线播放| 国产成人久久久| 欧美成人全部免费| 日韩精品免费综合视频在线播放| 欧美激情在线狂野欧美精品| 亚洲男人天堂古典| 最新的欧美黄色| 日韩电影大全免费观看2023年上| 青青在线视频一区二区三区| 久久久久久久久久久国产| 久久综合亚洲社区| 日韩av网站电影| 久久久精品免费视频| 欧美精品情趣视频| 尤物99国产成人精品视频| 国外成人在线播放| 92看片淫黄大片欧美看国产片| 亚洲免费一级电影| 日韩欧美中文第一页| 国产日韩精品电影| 在线看片第一页欧美| 日韩精品中文字幕有码专区| 日韩福利伦理影院免费| 欧美与欧洲交xxxx免费观看| 97在线看免费观看视频在线观看| 97av在线视频| 久久亚洲精品视频| 国产精品美乳一区二区免费| 日韩精品中文在线观看| 国产精品午夜视频| 亚洲视频免费一区| 久久久久免费精品国产| 国产视频久久久久| 亚洲www在线观看| 97精品欧美一区二区三区| 国内精品小视频在线观看| 国产精品三级久久久久久电影| 亚洲小视频在线| 日韩暖暖在线视频| 亚洲欧美日韩国产精品| 96sao精品视频在线观看| 亚洲xxxx在线| 欧美大荫蒂xxx| 成人国产精品日本在线| 欧美黄色性视频| 欧美精品xxx| 日本高清+成人网在线观看| 亚洲欧美日韩国产精品| 亚洲人成亚洲人成在线观看| 国产午夜精品全部视频播放| 日本中文字幕成人| 欧美日韩高清区| 亚洲国产精品电影在线观看| 一区二区三区四区在线观看视频| 亚洲精品视频在线播放| 欧洲亚洲女同hd| 国产成人91久久精品| 国产精品免费久久久久久| 亚洲天堂av网| 日韩精品在线影院| 欧美综合在线观看| 亚洲国产古装精品网站| 国产精品观看在线亚洲人成网| 亚洲男人天堂古典| 91理论片午午论夜理片久久| 色婷婷综合成人av| 91在线直播亚洲| 亚洲激情在线视频| 法国裸体一区二区| 成人在线中文字幕| 亚洲精品乱码久久久久久金桔影视| 国产精品美女呻吟| 亚洲久久久久久久久久久| 91精品国产自产在线| 欧美极度另类性三渗透| 国产精品久久视频| 九九综合九九综合| 亚洲一区中文字幕在线观看| 亚洲午夜未满十八勿入免费观看全集| 岛国av一区二区三区| 最新国产成人av网站网址麻豆| 亚洲人成绝费网站色www| 亚洲国产精品99久久| 欧美电影免费观看网站| 欧美一乱一性一交一视频| 久久综合伊人77777| 欧美野外猛男的大粗鳮| 国产精品999999| 91精品视频在线播放| 欧美日韩中文字幕日韩欧美| 国产精品久久久久高潮| 国产亚洲欧美另类中文| 欧美男插女视频| 亚洲第一区在线观看| 亚洲国产精品久久| 久久亚洲精品中文字幕冲田杏梨| 久久精品国产成人精品| 色偷偷88888欧美精品久久久| 久久精品欧美视频| 亚洲精品日韩在线| 日韩av网站电影| 国产精品日日摸夜夜添夜夜av| 中文字幕亚洲第一| 欧美人在线观看| 国自产精品手机在线观看视频| 国产成人一区二区| 91国内揄拍国内精品对白| 国产精品久久久久久久久久三级| 亚洲最新在线视频| 亚洲自拍高清视频网站| 亚洲一区二区久久久久久| 国产日韩在线一区| 国产精品久久久久久久久免费看| 91精品国产高清久久久久久| 亚洲精品有码在线| 日韩视频在线观看免费| 欧美日韩高清在线观看| 国产性色av一区二区| 91免费综合在线| 国产成人精彩在线视频九色| 国产亚洲激情在线| 欧美成aaa人片在线观看蜜臀| 国产精品免费视频xxxx| 亚洲精品影视在线观看| 国产精品日韩欧美| 日韩国产高清视频在线| 欧美电影《睫毛膏》| 亚洲欧洲免费视频| 亚洲大胆人体在线| 国产乱人伦真实精品视频| 亚洲男人av电影|