问题描述
如何继承/扩展使用揭示原型模式的类? 有没有办法使private变量和函数protected?
示例基本对象:
myNameSpace.Person = function() { this.name= ""; this.id = 0; }; myNameSpace.Person.prototype = function(){ var foo = function(){ //sample private function }; var loadFromJSON = function (p_jsonObject) { ... }; var toJSON = function () { ... }; var clone = function (p_other) { ... }; return { loadFromJSON : loadFromJSON, toJSON: toJSON, clone: clone }; }();
推荐答案
JavaScript中没有受保护的变量/属性.但是,当您在同一范围内声明继承类时,您可以重复使用"私有"变量,当您的情况下,这似乎是可能的,当私人变量仅是原型的"隐藏实用程序"时.
MyNamespace.Person = function Person(params) { // private variables and functions, individual for each Person instance var anything, id; function execute_something() {} // public properties: this.name = ""; this.getId = function getId(){ // called a "privileged function", because it has access to private variables } } MyNamespace.American = function(params) { MyNamespace.Person.call(this, params); // inherit name and getId() } (function() { // new scope for // hidden utility functions and other private things function foo() { } function helpJSON() { } function fromJSON() { } var bar; (function(personProto) { // new scope for prototype module (not explicitly needed) // "private" /static/ variables (and functions, if you want them private) var personCount = 0; personProto.clone = function clone() { return this.constructor(myself); // or something }; personProto.toJSON = function toJSON() { // use of helpJSON() }; personProto.fromJSON = fromJSON; // direct use })(MyNamespace.Person.prototype); (function(amiProto) { // just the same as above, if needed amiProto.special = function() { // use foo() and co }; })( MyNamespace.American.prototype = Object.create(MyNamespace.Person.prototype) ); })();
这是继承的JavaScript方式,这意味着美国人的原型继承了clone(),tojson()()和fromjson()(从人的原型中自动函数.当然要覆盖.该功能是
new MyNamespace.American() instanceof MyNamespace.Person; // true
当然,如果您不需要,并且想要使用更类似模块的方式,则可以重复使用实用程序功能,即只需复制它们:
(function() { // hidden utility functions and other private things var bar; var personCount; function foo() { } function helpJSON() { } function fromJSON() { } function clone() { return this.constructor(myself); // or something } function toJSON() { } (function(personProto) { // new scope, not really needed // private variables are useless in here personProto.clone = clone; personProto.toJSON = toJSON; personProto.fromJSON = fromJSON; })(MyNamespace.Person.prototype); (function(amiProto) { // new scope, not really needed // copied from personProto amiProto.clone = clone; amiProto.toJSON = toJSON; amiProto.fromJSON = fromJSON; // and now the differences amiProto.special = function() { // use foo() and co }; })(MyNamespace.American.prototype); })();
问题描述
How do I inherit/extend classes that are using the Revealing Prototype pattern? And is there a way to make the private variables and functions protected?
Example base object:
myNameSpace.Person = function() { this.name= ""; this.id = 0; }; myNameSpace.Person.prototype = function(){ var foo = function(){ //sample private function }; var loadFromJSON = function (p_jsonObject) { ... }; var toJSON = function () { ... }; var clone = function (p_other) { ... }; return { loadFromJSON : loadFromJSON, toJSON: toJSON, clone: clone }; }();
推荐答案
There are no protected variables/properties in JavaScript. Though, you can reuse "private" variables when you declare the inheriting classes in the same scope, which seems possible in your case when the private variables are only "hidden utilities" of your prototype.
MyNamespace.Person = function Person(params) { // private variables and functions, individual for each Person instance var anything, id; function execute_something() {} // public properties: this.name = ""; this.getId = function getId(){ // called a "privileged function", because it has access to private variables } } MyNamespace.American = function(params) { MyNamespace.Person.call(this, params); // inherit name and getId() } (function() { // new scope for // hidden utility functions and other private things function foo() { } function helpJSON() { } function fromJSON() { } var bar; (function(personProto) { // new scope for prototype module (not explicitly needed) // "private" /static/ variables (and functions, if you want them private) var personCount = 0; personProto.clone = function clone() { return this.constructor(myself); // or something }; personProto.toJSON = function toJSON() { // use of helpJSON() }; personProto.fromJSON = fromJSON; // direct use })(MyNamespace.Person.prototype); (function(amiProto) { // just the same as above, if needed amiProto.special = function() { // use foo() and co }; })( MyNamespace.American.prototype = Object.create(MyNamespace.Person.prototype) ); })();
This is the JavaScript way of inheritance, which means American's prototype inherits the clone(), toJSON() and fromJSON() functions automagically from the Person's prototype. Of course overwritable. And the feature is
new MyNamespace.American() instanceof MyNamespace.Person; // true
Of course, if you don't need that, and want use the more module-like way, you could reuse the utility functions, i.e. just copy them:
(function() { // hidden utility functions and other private things var bar; var personCount; function foo() { } function helpJSON() { } function fromJSON() { } function clone() { return this.constructor(myself); // or something } function toJSON() { } (function(personProto) { // new scope, not really needed // private variables are useless in here personProto.clone = clone; personProto.toJSON = toJSON; personProto.fromJSON = fromJSON; })(MyNamespace.Person.prototype); (function(amiProto) { // new scope, not really needed // copied from personProto amiProto.clone = clone; amiProto.toJSON = toJSON; amiProto.fromJSON = fromJSON; // and now the differences amiProto.special = function() { // use foo() and co }; })(MyNamespace.American.prototype); })();