私のコードの一部がスタックオーバーフローを引き起こしている理由を理解するのに役立つことを望んでいました。
問題のコード:
var ClassCreator = {
create: function(class_object,ParentClass){
var created_class = null;
created_class = function(){
if(arguments.length == 0){
this.constructor();
}else{
this.constructor.apply(this,arguments);
}
};
this._grantInheritance(created_class,ParentClass);
this._grantMethods(created_class,class_object);
return created_class;
},
_grantInheritance: function(created_class,ParentClass){
if(ParentClass){
created_class.prototype = ParentClass.prototype;
created_class.prototype.BaseClass = ParentClass;
}
},
_grantMethods: function(created_class,creation_object){
//If there's no constructor provided, add a default constructor.
if(!creation_object.constructor){
creation_object.prototype.constructor = function(){};
}
//Add the creation_object's methods to the class we're creating.
for(var property in creation_object){
created_class.prototype[property] = creation_object[property];
}
}
};
var SuperSuperObject = ClassCreator.create({
constructor: function(){
document.write("Hello");
}
});
var SuperObject = ClassCreator.create({
constructor: function(){
this.BaseClass.call(this);
document.write(" ");
}
},SuperSuperObject);
var RegularObject = ClassCreator.create({
constructor: function(){
this.BaseClass.call(this);
document.write(" World");
}
},SuperObject);
var test = new RegularObject();
私が理解できる限り、RegularObjectsコンストラクターでthis.BaseClass.callを呼び出すと、RegularObjectsコンストラクターを再度呼び出そうとするため、スタックオーバーフローが発生します。SuperObjectのコンストラクターではなく、RegularObjectのコンストラクターを呼び出す理由はわかりません。何か案は?
編集:将来誰かがそれを望む場合に備えて、私の解決策:
var ClassCreator = {
__PROTOTYPE_CONSTRUCTOR_SIGNAL__: "1821fe18a870e71b29a6219e076b80bb",
create: function(class_object,ParentClass){
var created_class = null;
created_class = function(){
var call_class = null;
if(arguments.length == 1){
if(arguments[0] == ClassCreator.__PROTOTYPE_CONSTRUCTOR_SIGNAL__){
if(this.prototypeConstructor){
this.prototypeConstructor();
}
return;
}
}
if(!this.__construct_stack){
this.__construct_stack = 0;
}
call_class = this;
for(var counter = 0;counter<this.__construct_stack;counter++){
call_class = call_class.BaseClass.prototype;
}
this.__construct_stack++;
if(arguments.length == 0){
call_class.constructor.call(this);
}else{
call_class.constructor.apply(this,arguments);
}
return this;
};
this._grantInheritance(created_class,ParentClass);
this._grantMethods(created_class,class_object);
return created_class;
},
_grantInheritance: function(created_class,ParentClass){
if(ParentClass){
created_class.prototype = new ParentClass(this.__PROTOTYPE_CONSTRUCTOR_SIGNAL__);
created_class.prototype.BaseClass = ParentClass;
}
},
_grantMethods: function(created_class,creation_object){
//If there's no constructor provided, add a default constructor.
if(!creation_object.constructor){
creation_object.prototype.constructor = function(){};
}
//Add the creation_object's methods to the class we're creating.
for(var property in creation_object){
created_class.prototype[property] = creation_object[property];
}
}
};