35

如您所知,我们可以在 JS 中使用defineProperty(). 尝试使用defineProperty().

这是一个示例代码:

我有一个必须添加到对象的字段数组

fields = ["id", "name", "last_login"]

我也有一堂课会被修改

var User = (function(){
    // constructor
    function User(id, name){
        this.id     = id
        this.name   = name
    }
    return User;
})();

以及一个将字段添加到类的函数defineProperty()

var define_fields = function (fields){
    fields.forEach(function(field_name){
        var value = null
        Object.defineProperty(User.prototype, field_name, {
            get: function(){ return value }
            set: function(new_value){
                /* some business logic goes here */
                value = new_value
            }
        })
    })
};

运行后,define_fields()我在实例中有我的字段User

define_fields(fields);
user1 = new User(1, "Thomas")
user2 = new User(2, "John")

但是这些属性的值是相同的

console.log(user2.id, user2.name) // 2, John
console.log(user1.id, user1.name) // 2, John

defineProperty()在这种情况下有什么办法可以正常工作吗?如果我理解问题是value哪个类的每个实例都相同,但我不知道如何解决它。提前感谢您的回答。

UPD: 这种方式抛出“RangeError:超出最大调用堆栈大小”

var define_fields = function (fields){
    fields.forEach(function(field_name){
        Object.defineProperty(User.prototype, field_name, {
            get: function(){ return this[field_name] }
            set: function(new_value){
                /* some business logic goes here */
                this[field_name] = new_value
            }
        })
    })
};
4

6 回答 6

58

请不要实现任何其他版本,因为它会占用您应用程序中的所有内存:

var Player = function(){this.__gold = 0};

Player.prototype = {

    get gold(){
        return this.__gold * 2;
    },



    set gold(gold){
        this.__gold = gold;
    },
};

var p = new Player();
p.gold = 2;
alert(p.gold); // 4

如果实例化 10000 个对象:

  • 使用我的方法:内存中只有两个函数;
  • 使用其他方法:10000 * 2 = 20000 个函数在内存中;
于 2014-10-03T14:48:26.690 回答
18

在他回答三分钟后,我得出了与Mikhail Kraynov相同的结论。每次调用构造函数时,该解决方案都会定义新属性。我想知道,正如你所问的,是否有一种方法可以将 getter 和 setter 放入原型中。这是我想出的:

var User = (function () {
  function User (id, nam) {
    Object.defineProperty (this, '__',  // Define property for field values   
       { value: {} });

    this.id = id;
    this.nam = nam;
  }

  (function define_fields (fields){
    fields.forEach (function (field_name) {
      Object.defineProperty (User.prototype, field_name, {
        get: function () { return this.__ [field_name]; },
        set: function (new_value) {
               // some business logic goes here 
               this.__[field_name] = new_value;
             }
      });
    });
  }) (fields);

  return User;
}) ();  

在这个解决方案中,我在原型中定义了字段 getter 和 setter,但在每个保存字段值的实例中引用了一个(隐藏的)属性。

在这里查看小提琴:http: //jsfiddle.net/Ca7yq

我在小提琴中添加了更多代码以显示对属性枚举的一些影响:http: //jsfiddle.net/Ca7yq/1/

于 2012-12-27T11:18:04.597 回答
8

在我看来,当您为原型定义属性时,所有实例都共享该属性。所以正确的变体可能是

var User = (function(){
// constructor
function User(id, name){
    this.id     = id
    this.name   = name

    Object.defineProperty(this, "name", {
        get: function(){ return name },
        set: function(new_value){
            //Some business logic, upperCase, for example
            new_value = new_value.toUpperCase();
            name = new_value
        }
    })
}
return User;
})();
于 2012-12-27T09:40:21.830 回答
5

当您在所有用户实例的原型对象上定义属性时,所有这些对象将共享同一个value变量。如果这不是您想要的,您将需要分别调用defineFields每个用户实例 - 在构造函数中:

function User(id, name){
    this.define_fields(["name", "id"]);
    this.id     = id
    this.name   = name
}
User.prototype.define_fields = function(fields) {
    var user = this;
    fields.forEach(function(field_name) {
        var value;
        Object.defineProperty(user, field_name, {
            get: function(){ return value; },
            set: function(new_value){
                /* some business logic goes here */
                value = new_value;
            }
        });
    });
};
于 2012-12-27T11:16:49.417 回答
1

此解决方案没有额外的内存消耗。您更新的代码已接近。你只需要使用 this.props[field_name] 而不是直接使用 this[field_name]。

请注意,defineProperty 调用替换为 Object.create

Js 小提琴http://jsfiddle.net/amuzalevskyi/65hnpad8/

// util
function createFieldDeclaration(fields) {
    var decl = {};
    for (var i = 0; i < fields.length; i++) {
        (function(fieldName) {
            decl[fieldName] = {
                get: function () {
                    return this.props[fieldName];
                },
                set: function (value) {
                    this.props[fieldName] = value;
                }
            }
        })(fields[i]);
    }
    return decl;
}

// class definition
function User(id, name) {
    this.props = {};
    this.id = id;
    this.name = name;
}
User.prototype = Object.create(Object.prototype, createFieldDeclaration(['id','name']));

// tests
var Alex = new User(0, 'Alex'),
    Andrey = new User(1, 'Andrey');

document.write(Alex.name + '<br/>'); // Alex
document.write(Andrey.name + '<br/>'); // Andrey

Alex.name = "Alexander";
document.write(Alex.name + '<br/>'); // Alexander
document.write(Andrey.name + '<br/>'); //Andrey
于 2014-11-29T19:53:29.727 回答
0

从接受的答案中,我意识到我们在这里要做的是定义私有实例变量。这些变量应该在实例(this)上,而不是在原型对象上。通常我们通过在属性名称前加上下划线来命名私有变量。

var Vehicle = {};
Object.defineProperty(Vehicle, "make", {
    get: function() { return this._make; }
    set: function(value) { this._make = value; }
});

function Car(m) { this.make = m; }    //this will set the private var _make
Car.prototype = Vehicle;

接受的答案基本上将所有私有变量放在一个容器中,这实际上更好。

于 2014-10-21T17:05:21.800 回答