- 论坛徽章:
- 0
|
【构造函数方式】
这种方式是为了解决上面工厂函数的第一个问题,即没有new运算符的问题。可是第二个问题它依然不能解决。我们来看看。- function Obj(name) {
- this.name = name;
- this.showName = function () {
- alert(this.name);
- }
- }
- var obj1 = new Obj('obj_one');
- var obj2 = new Obj('obj_two');
复制代码 它的好处是不用在构造函数内新建一个对象了,因为new运算符执行的时候会自动创建一个对象,并且只有通过this才能访问这个对象。所以我们可以直接通过this来对这个对象进行赋值。而且不用再return,因为this指向默认为构造函数的返回值。
同时,用了new关键字来创建我们想要的对象是不是感觉更“正式”了。
可惜,它仍然不能解决会重复生成方法函数的问题,这个情况和工厂函数一样。
【原型方式】
这种方式对比以上方式,有个很大的优势,就是它解决了方法函数会被生成多次的问题。它利用了对象的prototype属性。我们依赖原型可以重写对象实例。- var Obj = function () {}
- Obj.prototype.name = 'me';
- Obj.prototype.showName = function () {
- alert(this.name);
- }
- var obj1 = new Obj();
- var obj2 = new Obj();
复制代码 我们依赖原型对构造函数进行重写,无论是属性还是方法都是通过原型引用的方式给新建的对象,因此都只会被创建一次。可惜的是,这种方式存在两个致命的问题:
1。没办法在构建对象的时候就写入想要的属性,因为原型在构造函数作用域外边,没办法通过传递参数的方式在对象创建的时候就写入属性值。只能在对象创建完毕后对值进行重写。
2。致命问题在于当属性指向对象时,这个对象会被多个实例所共享。考虑下面的代码:- var Obj = function () {}
- Obj.prototype.name = 'me';
- Obj.prototype.flag = new Array('A', 'B');
- Obj.prototype.showName = function () {
- alert(this.name);
- }
- var obj1 = new Obj();
- var obj2 = new Obj();
- obj1.flag.push('C');
- alert(obj1.flag); // A,B,C
- alert(obj2.flag); //A,B,C
复制代码 是的,当flag属性指向对象时,那么实例obj1和obj2都共享它,哪怕我们仅仅改变了obj1的flag属性,但是它的改变在实例obj2中任然可见。
面对这个问题,让我们不得不想是否应该把【构造函数方式】和【原型方式】结合起来,让他们互补。。。
【构造函数和原型混合方式】
我们让属性用构造函数方式创建,方法用原型方式创建即可:- var Obj = function (name) {
- this.name = name;
- this.flag = new Array('A', 'B');
- }
- Obj.prototype = {
- showName : function () {
- alert(this.name);
- }
- }
- var obj1 = new Obj();
- var obj2 = new Obj();
- obj1.flag.push('C');
- alert(obj1.flag); // A,B,C
- alert(obj2.flag); //A,B
复制代码 这种方式有效地结合了原型和构造函数的优势,是目前用的最多,也是副作用最少的方式。
不过,有些追求完美的家伙还不满足,因为在视觉上还没达到他们的要求,因为通过原型来创建方法的过程在视觉上还是会让人觉得它不太像实例的方法(尤其对于传统OOP语言的开发者来说。)
所以,我们可以让原型活动起来,让他也加入到构造函数里面去,好让这个构造函数在视觉上更为统一。而这一系列的过程只需用一个判断即可完成。- var Obj = function (name) {
- this.name = name;
- this.flag = new Array('A', 'B');
- if (typeof Obj._init == 'undefined') {
- Obj.prototype = {
- showName : function () {
- alert(this.name);
- }
- };
- Obj._init = true;
- }
- }
复制代码 如上,用_init作为一个标志来判断是否已经给原型创建了方法。如果是那么就不再执行。这样其实在本质上是没有任何变化的,方法仍是通过原型创建,唯一的区别在于这个构造函数看起来“江山统一”了。
但是这种动态原型的方式是有问题的,《JavaScript高级程序设计》里并没有深究。创建第一个对象的时候会因为prototype在对象实例化之前没来的及建起来,是根本无法访问的。所以第一个对象是无法访问原型方法的。同时这种方式在子类继承中也会有问题。 |
|