溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Javascript中怎么實(shí)現(xiàn)面向切面編程

發(fā)布時(shí)間:2021-06-17 15:59:30 來(lái)源:億速云 閱讀:202 作者:Leah 欄目:web開發(fā)

本篇文章為大家展示了Javascript中怎么實(shí)現(xiàn)面向切面編程,內(nèi)容簡(jiǎn)明扼要并且容易理解,絕對(duì)能使你眼前一亮,通過這篇文章的詳細(xì)介紹希望你能有所收獲。

面向切面編程(Aspect-oriented programming,AOP)是一種編程范式。做后端 Java web 的同學(xué),特別是用過 Spring 的同學(xué)肯定對(duì)它非常熟悉。AOP 是 Spring 框架里面其中一個(gè)重要概念。可是在 Javascript 中,AOP 是一個(gè)經(jīng)常被忽視的技術(shù)點(diǎn)。

場(chǎng)景

假設(shè)你現(xiàn)在有一個(gè)牛逼的日歷彈窗,有一天,老板讓你統(tǒng)計(jì)一下每天這個(gè)彈窗里面某個(gè)按鈕的點(diǎn)擊數(shù),于是你在彈窗里做了埋點(diǎn);

過了一個(gè)星期,老板說用戶反饋這個(gè)彈窗好慢,各種卡頓。你想看一下某個(gè)函數(shù)的平均執(zhí)行時(shí)間,于是你又在彈窗里加上了性能統(tǒng)計(jì)代碼。

時(shí)間久了,你會(huì)發(fā)現(xiàn)你的業(yè)務(wù)邏輯里包含了大量的和業(yè)務(wù)無(wú)關(guān)的東西,即使是一些你已經(jīng)封裝過的函數(shù)。
那么 AOP 就是為了解決這類問題而存在的。

關(guān)注點(diǎn)分離

分離業(yè)務(wù)代碼和數(shù)據(jù)統(tǒng)計(jì)代碼(非業(yè)務(wù)代碼),無(wú)論在什么語(yǔ)言中,都是AOP的經(jīng)典應(yīng)用之一。從核心關(guān)注點(diǎn)中分離出橫切關(guān)注點(diǎn),是 AOP 的核心概念。
在前端的常見需求中,有以下一些業(yè)務(wù)可以使用 AOP 將其從核心關(guān)注點(diǎn)中分離出來(lái)

  1. Node.js 日志log

  2. 埋點(diǎn)、數(shù)據(jù)上報(bào)

  3. 性能分析、統(tǒng)計(jì)函數(shù)執(zhí)行時(shí)間

  4. 給ajax請(qǐng)求動(dòng)態(tài)添加參數(shù)、動(dòng)態(tài)改變函數(shù)參數(shù)

  5. 分離表單請(qǐng)求和驗(yàn)證

  6. 防抖與節(jié)流

 裝飾器(Decorator)

提到 AOP 就要說到裝飾器模式,AOP 經(jīng)常會(huì)和裝飾器模式混為一談。

在ES6之前,要使用裝飾器模式,通常通過Function.prototype.before做前置裝飾,和Function.prototype.after做后置裝飾(見《Javascript設(shè)計(jì)模式和開發(fā)實(shí)踐》)。

Javascript 引入的 Decorator ,和 Java 的注解在語(yǔ)法上很類似,不過在語(yǔ)義上沒有一丁點(diǎn)關(guān)系。Decorator 提案提供了對(duì) Javascript 的類和類里的方法進(jìn)行裝飾的能力。(盡管只是在編譯時(shí)運(yùn)行的函數(shù)語(yǔ)法糖)

埋點(diǎn)數(shù)據(jù)上報(bào)

因?yàn)樵谑褂?React 的實(shí)際開發(fā)中有大量基于 Class 的 Component,所以我這里用 React 來(lái)舉例。
比如現(xiàn)在頁(yè)面中有一個(gè)button,點(diǎn)擊這個(gè)button會(huì)彈出一個(gè)彈窗,與此同時(shí)要進(jìn)行數(shù)據(jù)上報(bào),來(lái)統(tǒng)計(jì)有多少用戶點(diǎn)擊了這個(gè)登錄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裝飾器,他會(huì)修改這個(gè) Class 上的原型方法,下面是@send裝飾器的實(shí)現(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;
}

在按鈕點(diǎn)擊后執(zhí)行showDialog前,可以執(zhí)行我們想要的切面操作,我們可以將埋點(diǎn),數(shù)據(jù)上報(bào)相關(guān)代碼封裝在這個(gè)裝飾器里面來(lái)實(shí)現(xiàn) AOP。

前置裝飾和后置裝飾

上面的send這個(gè)裝飾器其實(shí)是一個(gè)前置裝飾器,我們可以將它再封裝一下使它可以前置執(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裝飾器在一個(gè)原型方法前切入任意的非業(yè)務(wù)代碼。

function beforeLog() {
  console.log(`before calling ${name} with`, arguments);
}
class Dialog {
  ...
  @before(beforeLog)
  showDialog(content) {
    // do things
  }
  ...
}

和@before裝飾器類似,可以實(shí)現(xiàn)一個(gè)@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;
  }
}

性能分析

有時(shí)候我們想統(tǒng)計(jì)一段代碼在用戶側(cè)的執(zhí)行時(shí)間,但是又不想將打點(diǎn)代碼嵌入到業(yè)務(wù)代碼中,同樣可以利用裝飾器來(lái)做 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)計(jì)執(zhí)行時(shí)間的類方法前面加上@measure就行了,這樣做性能統(tǒng)計(jì)的代碼就不會(huì)侵入到業(yè)務(wù)代碼中。

class Dialog {
  ...
  @measure
  showDialog(content) {
    // do things
  }
  ...
}

上述內(nèi)容就是Javascript中怎么實(shí)現(xiàn)面向切面編程,你們學(xué)到知識(shí)或技能了嗎?如果還想學(xué)到更多技能或者豐富自己的知識(shí)儲(chǔ)備,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI