詳解在Javascript中進行面向切面編程
面向切面編程(Aspect-oriented programming,AOP)是一種編程范式。做后端 Java web 的同學(xué),特別是用過 Spring 的同學(xué)肯定對它非常熟悉。AOP 是 Spring 框架里面其中一個重要概念??墒窃?Javascript 中,AOP 是一個經(jīng)常被忽視的技術(shù)點。
場景
假設(shè)你現(xiàn)在有一個牛逼的日歷彈窗,有一天,老板讓你統(tǒng)計一下每天這個彈窗里面某個按鈕的點擊數(shù),于是你在彈窗里做了埋點;
過了一個星期,老板說用戶反饋這個彈窗好慢,各種卡頓。你想看一下某個函數(shù)的平均執(zhí)行時間,于是你又在彈窗里加上了性能統(tǒng)計代碼。
時間久了,你會發(fā)現(xiàn)你的業(yè)務(wù)邏輯里包含了大量的和業(yè)務(wù)無關(guān)的東西,即使是一些你已經(jīng)封裝過的函數(shù)。
那么 AOP 就是為了解決這類問題而存在的。
關(guān)注點分離
分離業(yè)務(wù)代碼和數(shù)據(jù)統(tǒng)計代碼(非業(yè)務(wù)代碼),無論在什么語言中,都是AOP的經(jīng)典應(yīng)用之一。從核心關(guān)注點中分離出橫切關(guān)注點,是 AOP 的核心概念。
在前端的常見需求中,有以下一些業(yè)務(wù)可以使用 AOP 將其從核心關(guān)注點中分離出來
- Node.js 日志log
- 埋點、數(shù)據(jù)上報
- 性能分析、統(tǒng)計函數(shù)執(zhí)行時間
- 給ajax請求動態(tài)添加參數(shù)、動態(tài)改變函數(shù)參數(shù)
- 分離表單請求和驗證
- 防抖與節(jié)流
裝飾器(Decorator)
提到 AOP 就要說到裝飾器模式,AOP 經(jīng)常會和裝飾器模式混為一談。
在ES6之前,要使用裝飾器模式,通常通過Function.prototype.before做前置裝飾,和Function.prototype.after做后置裝飾(見《Javascript設(shè)計模式和開發(fā)實踐》)。
Javascript 引入的 Decorator ,和 Java 的注解在語法上很類似,不過在語義上沒有一丁點關(guān)系。Decorator 提案提供了對 Javascript 的類和類里的方法進行裝飾的能力。(盡管只是在編譯時運行的函數(shù)語法糖)
埋點數(shù)據(jù)上報
因為在使用 React 的實際開發(fā)中有大量基于 Class 的 Component,所以我這里用 React 來舉例。
比如現(xiàn)在頁面中有一個button,點擊這個button會彈出一個彈窗,與此同時要進行數(shù)據(jù)上報,來統(tǒng)計有多少用戶點擊了這個登錄button。
import React, { Component } from 'react';
import send from './send';
class Dialog extends Component {
constructor(props) {
super(props);
}
@send
showDialog(content) {
// do things
}
render() {
return (
<button onClick={() => this.showDialog('show dialog')}>showDialog</button>
)
}
}
export default Dialog;
上面代碼引用了@send裝飾器,他會修改這個 Class 上的原型方法,下面是@send裝飾器的實現(xiàn)
export default function send(target, name, descriptor) {
let oldValue = descriptor.value;
descriptor.value = function () {
console.log(`before calling ${name} with`, arguments);
return oldValue.apply(this, arguments);
};
return descriptor;
}
在按鈕點擊后執(zhí)行showDialog前,可以執(zhí)行我們想要的切面操作,我們可以將埋點,數(shù)據(jù)上報相關(guān)代碼封裝在這個裝飾器里面來實現(xiàn) AOP。
前置裝飾和后置裝飾
上面的send這個裝飾器其實是一個前置裝飾器,我們可以將它再封裝一下使它可以前置執(zhí)行任意函數(shù)。
function before(beforeFn = function () { }) {
return function (target, name, descriptor) {
let oldValue = descriptor.value;
descriptor.value = function () {
beforeFn.apply(this, arguments);
return oldValue.apply(this, arguments);
};
return descriptor;
}
}
這樣我們就可以使用@before裝飾器在一個原型方法前切入任意的非業(yè)務(wù)代碼。
function beforeLog() {
console.log(`before calling ${name} with`, arguments);
}
class Dialog {
...
@before(beforeLog)
showDialog(content) {
// do things
}
...
}
和@before裝飾器類似,可以實現(xiàn)一個@after后置裝飾器,只是函數(shù)的執(zhí)行順序不一樣。
function after(afterFn = function () { }) {
return function (target, name, descriptor) {
let oldValue = descriptor.value;
descriptor.value = function () {
let ret = oldValue.apply(this, arguments);
afterFn.apply(this, arguments);
return ret;
};
return descriptor;
}
}
性能分析
有時候我們想統(tǒng)計一段代碼在用戶側(cè)的執(zhí)行時間,但是又不想將打點代碼嵌入到業(yè)務(wù)代碼中,同樣可以利用裝飾器來做 AOP。
function measure(target, name, descriptor) {
let oldValue = descriptor.value;
descriptor.value = function () {
let ret = oldValue.apply(this, arguments);
performance.mark("startWork");
afterFn.apply(this, arguments);
performance.mark("endWork");
performance.measure("work", "startWork", "endWork");
performance
.getEntries()
.map(entry => JSON.stringify(entry, null, 2))
.forEach(json => console.log(json));
return ret;
};
return descriptor;
}
在要統(tǒng)計執(zhí)行時間的類方法前面加上@measure就行了,這樣做性能統(tǒng)計的代碼就不會侵入到業(yè)務(wù)代碼中。
class Dialog {
...
@measure
showDialog(content) {
// do things
}
...
}
小結(jié)
面向切面編程的重點就是將核心關(guān)注面分離出橫切關(guān)注面,前端可以用 AOP 優(yōu)雅的來組織數(shù)據(jù)上報、性能分析、統(tǒng)計函數(shù)的執(zhí)行時間、動態(tài)改變函數(shù)參數(shù)、插件式的表單驗證等代碼。
以上所述是小編給大家介紹的Javascript面向切面編程詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
css3元素簡單的閃爍效果實現(xiàn)(html5 jquery)
本篇文章主要介紹了css3元素簡單的閃爍效果實現(xiàn)(html5 jquery) 需要的朋友可以過來參考下,希望對大家有所幫助2013-12-12
JavaScript中的object轉(zhuǎn)換函數(shù)toString()與valueOf()介紹
這篇文章主要介紹了JavaScript中的object轉(zhuǎn)換函數(shù)toString()與valueOf()介紹,需要的朋友可以參考下2014-12-12

