衍生內建建構器的子類別 subclass
Subclassing a built-in 的兩個障礙:
Obstacle 1 帶有內部特性的實體
1 | function Super(x, y){ |
內建建構器會忽略傳入傳入作為this的子實體
[[Primitive Value]]
valueOf() -> toString()
內部實體特性 | |
---|---|
包裹器建構器 | |
Boolean | [[PrimitiveValue]] |
Number | [[PrimitiveValue]] |
String | [[PrimitiveValue]], [[GetOwnProperty]], length |
其他建構器 | |
Array | [[DefineOwnProperty]] |
Date | [[PrimitiveValue]] |
Function | [[Call]] |
RegExp | [[Match]] |
- Error, Object 沒有內部特性
解決方式
1 | function copyOwnPropertiesFrom(targetObj, source){ |
Obstacle 2 無法被當作函式呼叫的建構器
Ex. Error 無法用上述方式subclass他們
解決方式
- 子建構器Sub 內創建新的超實體Super 並複製自有特性到子實體Sub
1 | function NewError(){ |
- 代理 delegation
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33function MyAry(/*arguments*/){
this.ary = [];
Array.prototype.push.apply(this.ary, arguments);
}
Object.defineProperties(MyAry.prototype, {
size: {
get: function(){
var size = 0;
for (var i = 0; i < this.length; i++){
if(i in this) size++;
}
return size;
}
},
});
//限制: 不能使用[]取值 需用此方式設定get set達成這個功能
MyAry.prototype.get = function(index){
return this.ary[index];
}
MyAry.prototype.set = function(index, value){
return this.ary[index] = value;
}
//轉移所有Array.prototype的方法
Object.getOwnPropertyNames(Array.prototype).forEach(function(propKey){
MyAry.prototype[propKey] = function(){
return Array.prototype[propKey].apply(this.ary, arguments);
}
});
衍生內建建構器的子類別 subclass