JS裝飾器函數(shù)用法總結(jié)
在 ES6 中增加了對類對象的相關(guān)定義和操作(比如 class 和 extends ),這就使得我們在多個不同類之間共享或者擴展一些方法或者行為的時候,變得并不是那么優(yōu)雅。這個時候,我們就需要一種更優(yōu)雅的方法來幫助我們完成這些事情。
什么是裝飾器
Python 的裝飾器
在面向?qū)ο螅∣OP)的設(shè)計模式中,decorator被稱為裝飾模式。OOP的裝飾模式需要通過繼承和組合來實現(xiàn),而Python除了能支持 OOP 的 decorator 外,直接從語法層次支持 decorator。
如果你熟悉 python 的話,對它一定不會陌生。那么我們先來看一下 python 里的裝飾器是什么樣子的吧:
def decorator(f): print "my decorator" return f @decorator def myfunc(): print "my function" myfunc() # my decorator # my function
這里的 @decorator 就是我們說的裝飾器。在上面的代碼中,我們利用裝飾器給我們的目標方法執(zhí)行前打印出了一行文本,并且并沒有對原方法做任何的修改。代碼基本等同于:
def decorator(f):
def wrapper():
print "my decorator"
return f()
return wrapper
def myfunc():
print "my function"
myfunc = decorator(myfuc)
通過代碼我們也不難看出,裝飾器 decorator 接收一個參數(shù),也就是我們被裝飾的目標方法,處理完擴展的內(nèi)容以后再返回一個方法,供以后調(diào)用,同時也失去了對原方法對象的訪問。當我們對某個應(yīng)用了裝飾以后,其實就改變了被裝飾方法的入口引用,使其重新指向了裝飾器返回的方法的入口點,從而來實現(xiàn)我們對原函數(shù)的擴展、修改等操作。
ES7 的裝飾器
ES7 中的 decorator 同樣借鑒了這個語法糖,不過依賴于 ES5 的 Object.defineProperty 方法 。
Object.defineProperty
Object.defineProperty() 方法會直接在一個對象上定義一個新屬性,或者修改一個對象的現(xiàn)有屬性, 并返回這個對象。
該方法允許精確添加或修改對象的屬性。通過賦值來添加的普通屬性會創(chuàng)建在屬性枚舉期間顯示的屬性(for...in 或 Object.keys 方法), 這些值可以被改變,也可以被刪除。這種方法允許這些額外的細節(jié)從默認值改變。默認情況下,使用 Object.defineProperty() 添加的屬性值是不可變的。
語法
Object.defineProperty(obj, prop, descriptor)
- obj:要在其上定義屬性的對象。
- prop:要定義或修改的屬性的名稱。
- descriptor:將被定義或修改的屬性描述符。
- 返回值:被傳遞給函數(shù)的對象。
在ES6中,由于 Symbol類型 的特殊性,用 Symbol類型 的值來做對象的key與常規(guī)的定義或修改不同,而Object.defineProperty 是定義 key為 Symbol 的屬性的方法之一。
屬性描述符
對象里目前存在的屬性描述符有兩種主要形式:數(shù)據(jù)描述符和存取描述符。
數(shù)據(jù)描述符是一個具有值的屬性,該值可能是可寫的,也可能不是可寫的。
- 存取描述符是由 getter-setter 函數(shù)對描述的屬性。
- 描述符必須是這兩種形式之一;不能同時是兩者。
數(shù)據(jù)描述符和存取描述符均具有以下可選鍵值:
configurable
當且僅當該屬性的 configurable 為 true 時,該屬性描述符才能夠被改變,同時該屬性也能從對應(yīng)的對象上被刪除。默認為 false。
enumerable
enumerable定義了對象的屬性是否可以在 for...in 循環(huán)和 Object.keys() 中被枚舉。
當且僅當該屬性的 enumerable 為 true 時,該屬性才能夠出現(xiàn)在對象的枚舉屬性中。默認為 false。
數(shù)據(jù)描述符同時具有以下可選鍵值:
value
該屬性對應(yīng)的值??梢允侨魏斡行У?JavaScript 值(數(shù)值,對象,函數(shù)等)。默認為 undefined。
writable
當且僅當該屬性的 writable 為 true 時,value 才能被賦值運算符改變。默認為 false。
存取描述符同時具有以下可選鍵值:
get
一個給屬性提供 getter 的方法,如果沒有 getter 則為 undefined。該方法返回值被用作屬性值。默認為 undefined。
set
一個給屬性提供 setter 的方法,如果沒有 setter 則為 undefined。該方法將接受唯一參數(shù),并將該參數(shù)的新值分配給該屬性。默認為 undefined。
如果一個描述符不具有value,writable,get 和 set 任意一個關(guān)鍵字,那么它將被認為是一個數(shù)據(jù)描述符。如果一個描述符同時有(value或writable)和(get或set)關(guān)鍵字,將會產(chǎn)生一個異常。
用法
類的裝飾
@testable
class MyTestableClass {
// ...
}
function testable(target) {
target.isTestable = true;
}
MyTestableClass.isTestable // true
上面代碼中,@testable 就是一個裝飾器。它修改了 MyTestableClass這 個類的行為,為它加上了靜態(tài)屬性isTestable。testable 函數(shù)的參數(shù) target 是 MyTestableClass 類本身。
基本上,裝飾器的行為就是下面這樣。
@decorator
class A {}
// 等同于
class A {}
A = decorator(A) || A;
也就是說,裝飾器是一個對類進行處理的函數(shù)。裝飾器函數(shù)的第一個參數(shù),就是所要裝飾的目標類。
如果覺得一個參數(shù)不夠用,可以在裝飾器外面再封裝一層函數(shù)。
function testable(isTestable) {
return function(target) {
target.isTestable = isTestable;
}
}
@testable(true)
class MyTestableClass {}
MyTestableClass.isTestable // true
@testable(false)
class MyClass {}
MyClass.isTestable // false
上面代碼中,裝飾器 testable 可以接受參數(shù),這就等于可以修改裝飾器的行為。
注意,裝飾器對類的行為的改變,是代碼編譯時發(fā)生的,而不是在運行時。這意味著,裝飾器能在編譯階段運行代碼。也就是說,裝飾器本質(zhì)就是編譯時執(zhí)行的函數(shù)。
前面的例子是為類添加一個靜態(tài)屬性,如果想添加實例屬性,可以通過目標類的 prototype 對象操作。
下面是另外一個例子。
// mixins.js
export function mixins(...list) {
return function (target) {
Object.assign(target.prototype, ...list)
}
}
// main.js
import { mixins } from './mixins'
const Foo = {
foo() { console.log('foo') }
};
@mixins(Foo)
class MyClass {}
let obj = new MyClass();
obj.foo() // 'foo'
上面代碼通過裝飾器 mixins,把Foo對象的方法添加到了 MyClass 的實例上面。
方法的裝飾
裝飾器不僅可以裝飾類,還可以裝飾類的屬性。
class Person {
@readonly
name() { return `${this.first} ${this.last}` }
}
上面代碼中,裝飾器 readonly 用來裝飾“類”的name方法。
裝飾器函數(shù) readonly 一共可以接受三個參數(shù)。
function readonly(target, name, descriptor){
// descriptor對象原來的值如下
// {
// value: specifiedFunction,
// enumerable: false,
// configurable: true,
// writable: true
// };
descriptor.writable = false;
return descriptor;
}
readonly(Person.prototype, 'name', descriptor);
// 類似于
Object.defineProperty(Person.prototype, 'name', descriptor);
- 裝飾器第一個參數(shù)是 類的原型對象,上例是 Person.prototype,裝飾器的本意是要“裝飾”類的實例,但是這個時候?qū)嵗€沒生成,所以只能去裝飾原型(這不同于類的裝飾,那種情況時target參數(shù)指的是類本身);
- 第二個參數(shù)是 所要裝飾的屬性名
- 第三個參數(shù)是 該屬性的描述對象
另外,上面代碼說明,裝飾器(readonly)會修改屬性的 描述對象(descriptor),然后被修改的描述對象再用來定義屬性。
函數(shù)方法的裝飾
裝飾器只能用于類和類的方法,不能用于函數(shù),因為存在函數(shù)提升。
另一方面,如果一定要裝飾函數(shù),可以采用高階函數(shù)的形式直接執(zhí)行。
function doSomething(name) {
console.log('Hello, ' + name);
}
function loggingDecorator(wrapped) {
return function() {
console.log('Starting');
const result = wrapped.apply(this, arguments);
console.log('Finished');
return result;
}
}
const wrapped = loggingDecorator(doSomething);
core-decorators.js
core-decorators.js是一個第三方模塊,提供了幾個常見的裝飾器,通過它可以更好地理解裝飾器。
@autobind
autobind 裝飾器使得方法中的this對象,綁定原始對象。
@readonly
readonly 裝飾器使得屬性或方法不可寫。
@override
override 裝飾器檢查子類的方法,是否正確覆蓋了父類的同名方法,如果不正確會報錯。
import { override } from 'core-decorators';
class Parent {
speak(first, second) {}
}
class Child extends Parent {
@override
speak() {}
// SyntaxError: Child#speak() does not properly override Parent#speak(first, second)
}
// or
class Child extends Parent {
@override
speaks() {}
// SyntaxError: No descriptor matching Child#speaks() was found on the prototype chain.
//
// Did you mean "speak"?
}
@deprecate (別名@deprecated)
deprecate 或 deprecated 裝飾器在控制臺顯示一條警告,表示該方法將廢除。
import { deprecate } from 'core-decorators';
class Person {
@deprecate
facepalm() {}
@deprecate('We stopped facepalming')
facepalmHard() {}
@deprecate('We stopped facepalming', { url: 'http://knowyourmeme.com/memes/facepalm' })
facepalmHarder() {}
}
let person = new Person();
person.facepalm();
// DEPRECATION Person#facepalm: This function will be removed in future versions.
person.facepalmHard();
// DEPRECATION Person#facepalmHard: We stopped facepalming
person.facepalmHarder();
// DEPRECATION Person#facepalmHarder: We stopped facepalming
//
// See http://knowyourmeme.com/memes/facepalm for more details.
//
@suppressWarnings
suppressWarnings 裝飾器抑制 deprecated 裝飾器導(dǎo)致的 console.warn() 調(diào)用。但是,異步代碼發(fā)出的調(diào)用除外。
使用場景
裝飾器有注釋的作用
@testable
class Person {
@readonly
@nonenumerable
name() { return `${this.first} ${this.last}` }
}
有了裝飾器,就可以改寫上面的代碼。裝飾
@connect(mapStateToProps, mapDispatchToProps)
export default class MyReactComponent extends React.Component {}
相對來說,后一種寫法看上去更容易理解。
新功能提醒或權(quán)限
菜單點擊時,進行事件攔截,若該菜單有新功能更新,則彈窗顯示。
/**
* @description 在點擊時,如果有新功能提醒,則彈窗顯示
* @param code 新功能的code
* @returns {function(*, *, *)}
*/
const checkRecommandFunc = (code) => (target, property, descriptor) => {
let desF = descriptor.value;
descriptor.value = function (...args) {
let recommandFuncModalData = SYSTEM.recommandFuncCodeMap[code];
if (recommandFuncModalData && recommandFuncModalData.id) {
setTimeout(() => {
this.props.dispatch({type: 'global/setRecommandFuncModalData', recommandFuncModalData});
}, 1000);
}
desF.apply(this, args);
};
return descriptor;
};
loading
在 React 項目中,我們可能需要在向后臺請求數(shù)據(jù)時,頁面出現(xiàn) loading 動畫。這個時候,你就可以使用裝飾器,優(yōu)雅地實現(xiàn)功能。
@autobind
@loadingWrap(true)
async handleSelect(params) {
await this.props.dispatch({
type: 'product_list/setQuerypParams',
querypParams: params
});
}
loadingWrap 函數(shù)如下:、
export function loadingWrap(needHide) {
const defaultLoading = (
<div className="toast-loading">
<Loading className="loading-icon"/>
<div>加載中...</div>
</div>
);
return function (target, property, descriptor) {
const raw = descriptor.value;
descriptor.value = function (...args) {
Toast.info(text || defaultLoading, 0, null, true);
const res = raw.apply(this, args);
if (needHide) {
if (get('finally')(res)) {
res.finally(() => {
Toast.hide();
});
} else {
Toast.hide();
}
}
};
return descriptor;
};
}
問題:這里大家可以想想看,如果我們不希望每次請求數(shù)據(jù)時都出現(xiàn) loading,而是要求只要后臺請求時間大于 300ms 時,才顯示loading,這里需要怎么改?
以上就是本次小編整理的關(guān)于JS裝飾器的相關(guān)知識點,感謝大家對腳本之家的支持。
相關(guān)文章
基于JavaScript實現(xiàn)繼承機制之調(diào)用call()與apply()的方法詳解
本文將介紹兩種很類似于對象冒充的繼承方式,即使用call()和apply()方法2013-05-05
使用?Next.js?Cli?快速搭建和運行?Web?應(yīng)用
這篇文章主要介紹了使用?Next.js?Cli?快速搭建和運行?Web?應(yīng)用,需要的朋友可以參考下2024-04-04
JavaScript中的Object對象學(xué)習(xí)教程
這篇文章主要介紹了JavaScript中的Object對象學(xué)習(xí)教程,是JavaScript入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下2016-05-05
JavaScript substr() 字符串截取函數(shù)使用詳解
substr() 方法可在字符串中抽取從 start 下標開始的指定數(shù)目的字符,文中配有大量實例代碼2013-04-04
ES6學(xué)習(xí)總結(jié)之Set和Map的使用
這篇博客介紹了 ES6 中的 Set 和 Map 數(shù)據(jù)結(jié)構(gòu),Set 是一個存儲唯一值的集合,支持添加、刪除、檢查元素的方法,Map 則是用于存儲鍵值對的集合,鍵和值都可以是任何類型,文章詳細講解了兩者的主要方法和用法,并與傳統(tǒng)的數(shù)組和對象進行了對比,突出 Set 和 Map 的獨特優(yōu)勢2024-08-08

