@@ -983,9 +983,9 @@ class Person {
983983
984984### core-decorators.js
985985
986- [core-decorators.js](https://github.com/jayphelps/core-decorators.js)提供了几个常见的修饰器。
986+ [core-decorators.js](https://github.com/jayphelps/core-decorators.js)是一个第三方模块, 提供了几个常见的修饰器,通过它可以更好地理解修饰器 。
987987
988- (1)@autobind
988+ ** (1)@autobind**
989989
990990autobind修饰器使得方法中的this对象,绑定原始对象。
991991
@@ -1006,7 +1006,7 @@ getPerson() === person;
10061006// true
10071007` ` `
10081008
1009- (2)@readonly
1009+ ** (2)@readonly**
10101010
10111011readonly修饰器是的属性或方法不可写。
10121012
@@ -1023,7 +1023,7 @@ dinner.entree = 'salmon';
10231023// Cannot assign to read only property 'entree' of [object Object]
10241024` ` `
10251025
1026- (3)@override
1026+ ** (3)@override**
10271027
10281028override修饰器检查子类的方法,是否正确覆盖了父类的同名方法,如果不正确会报错。
10291029
@@ -1051,7 +1051,7 @@ class Child extends Parent {
10511051}
10521052` ` `
10531053
1054- (4)@deprecate (别名@deprecated)
1054+ ** (4)@deprecate (别名@deprecated)**
10551055
10561056deprecate或deprecated修饰器在控制台显示一条警告,表示该方法将废除。
10571057
@@ -1084,7 +1084,7 @@ person.facepalmHarder();
10841084//
10851085` ` `
10861086
1087- (5)@suppressWarnings
1087+ ** (5)@suppressWarnings**
10881088
10891089suppressWarnings修饰器抑制decorated修饰器导致的` console .warn ()` 调用。但是,异步代码出发的调用除外。
10901090
@@ -1107,6 +1107,172 @@ person.facepalmWithoutWarning();
11071107// no warning is logged
11081108` ` `
11091109
1110+ ### Mixin
1111+
1112+ 在修饰器的基础上,可以实现Mixin模式。所谓Mixin模式,就是对象继承的一种替代方案,中文译为“混入”(mix in),意为在一个对象之中混入另外一个对象的方法。
1113+
1114+ 请看下面的例子。
1115+
1116+ ` ` ` javascript
1117+ const Foo = {
1118+ foo () { console .log (' foo' ) }
1119+ };
1120+
1121+ class MyClass {}
1122+
1123+ Object .assign (MyClass .prototype , Foo);
1124+
1125+ let obj = new MyClass ();
1126+ obj .foo () // 'foo'
1127+ ` ` `
1128+
1129+ 上面代码之中,对象Foo有一个foo方法,通过` Object .assign ` 方法,可以将foo方法“混入”MyClass类,导致MyClass的实例obj对象都具有foo方法。这就是“混入”模式的一个简单实现。
1130+
1131+ 下面,我们部署一个通用脚本` mixins .js ` ,将mixin写成一个修饰器。
1132+
1133+ ` ` ` javascript
1134+ export function mixins (... list ) {
1135+ return function (target ) {
1136+ Object .assign (target .prototype , ... list);
1137+ };
1138+ }
1139+ ` ` `
1140+
1141+ 然后,就可以使用上面这个修饰器,为类“混入”各种方法。
1142+
1143+ ` ` ` javascript
1144+ import { mixins } from ' ./mixins'
1145+
1146+ const Foo = {
1147+ foo () { console .log (' foo' ) }
1148+ };
1149+
1150+ @mixins (Foo)
1151+ class MyClass {}
1152+
1153+ let obj = new MyClass ();
1154+
1155+ obj .foo () // "foo"
1156+ ` ` `
1157+
1158+ 通过mixins这个修饰器,实现了在MyClass类上面“混入”Foo对象的foo方法。
1159+
1160+ ### Trait
1161+
1162+ Trait也是一种修饰器,功能与Mixin类型,但是提供更多功能,比如防止同名方法的冲突、排除混入某些方法、为混入的方法起别名等等。
1163+
1164+ 下面采用[traits-decorator](https://github.com/CocktailJS/traits-decorator)这个第三方模块作为例子。这个模块提供的traits修饰器,不仅可以接受对象,还可以接受ES6类作为参数。
1165+
1166+ ` ` ` javascript
1167+ import {traits } from ' traits-decorator'
1168+
1169+ class TFoo {
1170+ foo () { console .log (' foo' ) }
1171+ }
1172+
1173+ const TBar = {
1174+ bar () { console .log (' bar' ) }
1175+ }
1176+
1177+ @traits (TFoo, TBar)
1178+ class MyClass { }
1179+
1180+ let obj = new MyClass ()
1181+ obj .foo () // foo
1182+ obj .bar () // bar
1183+ ` ` `
1184+
1185+ 上面代码中,通过traits修饰器,在MyClass类上面“混入”了TFoo类的foo方法和TBar对象的bar方法。
1186+
1187+ Trait不允许“混入”同名方法。
1188+
1189+ ` ` ` javascript
1190+ import {traits } from ' traits-decorator'
1191+
1192+ class TFoo {
1193+ foo () { console .log (' foo' ) }
1194+ }
1195+
1196+ const TBar = {
1197+ bar () { console .log (' bar' ) },
1198+ foo () { console .log (' foo' ) }
1199+ }
1200+
1201+ @traits (TFoo, TBar)
1202+ class MyClass { }
1203+ // 报错
1204+ // throw new Error('Method named: ' + methodName + ' is defined twice.');
1205+ // ^
1206+ // Error: Method named: foo is defined twice.
1207+ ` ` `
1208+
1209+ 上面代码中,TFoo和TBar都有foo方法,结果traits修饰器报错。
1210+
1211+ 一种解决方法是排除TBar的foo方法。
1212+
1213+ ` ` ` javascript
1214+ import { traits , excludes } from ' traits-decorator'
1215+
1216+ class TFoo {
1217+ foo () { console .log (' foo' ) }
1218+ }
1219+
1220+ const TBar = {
1221+ bar () { console .log (' bar' ) },
1222+ foo () { console .log (' foo' ) }
1223+ }
1224+
1225+ @traits (TFoo, TBar:: excludes (' foo' ))
1226+ class MyClass { }
1227+
1228+ let obj = new MyClass ()
1229+ obj .foo () // foo
1230+ obj .bar () // bar
1231+ ` ` `
1232+
1233+ 上面代码使用绑定运算符(::)在TBar上排除foo方法,混入时就不会报错了。
1234+
1235+ 另一种方法是为TBar的foo方法起一个别名。
1236+
1237+ ` ` ` javascript
1238+ import { traits , alias } from ' traits-decorator'
1239+
1240+ class TFoo {
1241+ foo () { console .log (' foo' ) }
1242+ }
1243+
1244+ const TBar = {
1245+ bar () { console .log (' bar' ) },
1246+ foo () { console .log (' foo' ) }
1247+ }
1248+
1249+ @traits (TFoo, TBar:: alias ({foo: ' aliasFoo' }))
1250+ class MyClass { }
1251+
1252+ let obj = new MyClass ()
1253+ obj .foo () // foo
1254+ obj .aliasFoo () // foo
1255+ obj .bar () // bar
1256+ ` ` `
1257+
1258+ 上面代码为TBar的foo方法起了别名aliasFoo,于是MyClass也可以混入TBar的foo方法了。
1259+
1260+ alias和excludes方法,可以结合起来使用。
1261+
1262+ ` ` ` javascript
1263+ @traits (TExample:: excludes (' foo' ,' bar' ):: alias ({baz: ' exampleBaz' }))
1264+ class MyClass {}
1265+ ` ` `
1266+
1267+ 上面代码排除了TExample的foo方法和bar方法,为baz方法起了别名exampleBaz。
1268+
1269+ as方法则为上面的代码提供了另一种写法。
1270+
1271+ ` ` ` javascript
1272+ @traits (TExample:: as ({excludes: [' foo' , ' bar' ], alias: {baz: ' exampleBaz' }}))
1273+ class MyClass {}
1274+ ` ` `
1275+
11101276### Babel转码器的支持
11111277
11121278目前,Babel转码器已经支持Decorator,命令行的用法如下。
0 commit comments