欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JavaScript實(shí)現(xiàn)AOP詳解(面向切面編程,裝飾者模式)

 更新時(shí)間:2017年12月19日 09:50:34   作者:三十億少女的夢(mèng)  
下面小編就為大家分享一篇JavaScript實(shí)現(xiàn)AOP的方法(面向切面編程,裝飾者模式),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧

什么是AOP?

AOP(面向切面編程)的主要作用是把一些跟核心業(yè)務(wù)邏輯模塊無(wú)關(guān)的功能抽離出來(lái),這些跟業(yè)務(wù)邏輯無(wú)關(guān)的功能通常包括日志統(tǒng)計(jì)、安全控制、異常處理等。把這些功能抽離出來(lái)之后, 再通過(guò)“動(dòng)態(tài)織入”的方式摻入業(yè)務(wù)邏輯模塊中。

AOP能給我們帶來(lái)什么好處?

AOP的好處首先是可以保持業(yè)務(wù)邏輯模塊的純凈和高內(nèi)聚性,其次是可以很方便地復(fù)用日志統(tǒng)計(jì)等功能模塊。

JavaScript實(shí)現(xiàn)AOP的思路?

通常,在 JavaScript 中實(shí)現(xiàn) AOP,都是指把一個(gè)函數(shù)“動(dòng)態(tài)織入”到另外一個(gè)函數(shù)之中,具體的實(shí)現(xiàn)技術(shù)有很多,下面我用擴(kuò)展 Function.prototype 來(lái)做到這一點(diǎn)。請(qǐng)看下面代碼:

Function.prototype.before = function (beforefn) {
  var _self = this; //保存原函數(shù)引用
  return function () { //返回包含了原函數(shù)和新函數(shù)的"代理函數(shù)"
   beforefn.apply(this, arguments); //執(zhí)行新函數(shù),修正this
   return _self.apply(this, arguments); //執(zhí)行原函數(shù)
  }
 };
 Function.prototype.after = function (afterfn) {
  var _self = this;
  return function () {
   var ret = _self.apply(this, arguments);
   afterfn.apply(this, arguments);
   return ret;
  }
 };
 var func = function () {
  console.log("2")
 }
 func = func.before(function () {
  console.log("1");
 }).after(function () {
  console.log("3");
 } )
 func();

執(zhí)行結(jié)果如下:

我把負(fù)責(zé)打印數(shù)字1和打印數(shù)字3的兩個(gè)函數(shù)通過(guò)AOP的方式動(dòng)態(tài)植入func函數(shù)。通過(guò)執(zhí)行上面的代碼,我們看到控制臺(tái)順利地返回了執(zhí)行結(jié)果1、2、3。

這種使用AOP的方式來(lái)給函數(shù)添加職責(zé),也是JavaScript語(yǔ)言中的一種非常特別的巧妙的裝飾者模式實(shí)現(xiàn),下面我們來(lái)試試Function.prototype.before的威力,請(qǐng)看下面代碼:

Function.prototype.before = function (beforefn) {
  var __self = this; // 保存原函數(shù)的引用
  return function () { // 返回包含了原函數(shù)和新函數(shù)的"代理"函數(shù)
   beforefn.apply(this, arguments); // 執(zhí)行新函數(shù),且保證 this 不被劫持,新函數(shù)接受的參數(shù) // 也會(huì)被原封不動(dòng)地傳入原函數(shù),新函數(shù)在原函數(shù)之前執(zhí)行
   return __self.apply(this, arguments); // 執(zhí)行原函數(shù)并返回原函數(shù)的執(zhí)行結(jié)果, 2 // 并且保證 this 不被劫持
  }
 }
 Function.prototype.after = function (afterfn) {
  var __self = this;
  return function () {
   var ret = __self.apply(this, arguments);
   afterfn.apply(this, arguments);
   return ret;
  }
 };
 document.getElementById = document.getElementById.before(function(){ alert (1);
 });
 var button = document.getElementById( 'button' );

執(zhí)行結(jié)果:

我們給document.getElementById()做了一些裝飾,以后我們每次調(diào)用這個(gè)方法之前都會(huì)先執(zhí)行alert("1")這條語(yǔ)句,但是請(qǐng)注意我們這條語(yǔ)句并不是寫在了document.getElementById()這個(gè)方法的源碼中,而只是在他的外部給他加了裝飾,這樣帶來(lái)好處就是我們可以在不改變?cè)椒ǖ脑创a的情況下為他添加一些新的行為。國(guó)際慣例,舉個(gè)栗子:

我的同事寫了一個(gè)函數(shù)可以輸出當(dāng)前時(shí)間,而我現(xiàn)在的需求是輸出當(dāng)前天氣之后再輸出當(dāng)前時(shí)間,下面有兩種解決思路:

(1)傳統(tǒng)解決辦法: 拿同事的函數(shù)過(guò)來(lái),找到他輸出時(shí)間的代碼,在這些代碼之前加入輸出當(dāng)前天氣的代碼

(2)裝飾者模式解決辦法:拿同事的函數(shù)過(guò)來(lái),不用看他的源碼,直接給他的函數(shù)裝飾一下,裝飾的東西也就是輸出當(dāng)前天氣的代碼。

兩種方法都解決了問(wèn)題,但是他們的出發(fā)點(diǎn)是完全不同的:

(1)方法是改造原函數(shù)的內(nèi)部,我們就需要去理解源代碼,然后做修改。

(2)方法是給原函數(shù)添加了一層外套,我們根本不用管原本函數(shù)的內(nèi)部實(shí)現(xiàn)。

現(xiàn)在又有了新的需求:在輸出當(dāng)前時(shí)間之前,先輸出當(dāng)前溫度

(1)方法,我們?cè)诘谝粋€(gè)需求已經(jīng)把同事的代碼改的面目全非了,現(xiàn)在又要重新理解函數(shù)內(nèi)部,并加以修改(刪除輸出當(dāng)前天氣的代碼,然后加入輸出當(dāng)前溫度的代碼)。

(2)方法,同事原本的函數(shù)是沒(méi)有變的,我們現(xiàn)在給同事的函數(shù)換一件套(輸出當(dāng)前溫度)就可以了。

以上這篇JavaScript實(shí)現(xiàn)AOP詳解(面向切面編程,裝飾者模式)就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

最新評(píng)論