美文网首页让前端飞JavaScript 进阶营前端开发笔记
关于js(es5)如何优雅地创建对象

关于js(es5)如何优雅地创建对象

作者: SimpleCXD | 来源:发表于2018-11-15 00:32 被阅读5次

    一、前颜(yan)

    对象(Object Oriented,OO)对于基于类的语言来说是再普通不过的一个概念 了,比如C++,Java等等。
    而在es5(以下js指es5)中,由于没有类的概念,因此它的对象与其他基于类的语言的对象是不同的。
    因此在js中,关于对象的创建方法也有所不一样。
    本文介绍各种创建js对象的方法,以及优缺点。

    二、通过Object创建对象

    var cat = new Object();
    cat.name = 'kiki1';
    cat.age = 1;
    cat.speak = function () {
      console.log('Hello I am ', this.name);
    }
    

    缺点:当需要创建多个相似对象时,会产生大量的重复代码,比如这时候我想再创建一个cat2,这时就得重复一遍以上代码。

    var cat2 = new Object();
    cat2.name = 'kiki2';
    cat2.age = 2;
    cat2.speak = function () {
      console.log('Hello I am ', this.name);
    }
    

    三、通过对象字面量语法对象

    var cat = {
      name: 'kiki1',
      age: 1,
      speak: function () {
        console.log('Hello I am ', this.name);
      }
    }
    

    特点:通过此方法来创建对象相比Object创建对象代码简洁了很多
    缺点:与通过Object创建对象的缺点一致

    为了解决以上问题,以下模式来了。

    四、通过工厂模式创建对象

    function createCat(name, age) {
      var o = new Object();
      o.name = name;
      o.age = age;
      o.speak = function () {
        console.log('Hello I am ', this.name);
      }
      return o;
    }
    
    var cat1=createCat('kiki1',1);
    var cat2=createCat('kiki2',2);
    
    cat1.speak(); //Hello I am  kiki1
    cat2.speak(); //Hello I am  kiki2
    

    特点:通过工厂模式创建对象,可以解决创建多个相似对象的问题。
    缺点:无法识别对象,即无法知道一个对象的类型。

    五、通过构造函数模式创建对象

    通过构造函数模式创建对象,能够很好的解决工厂模式创建对象的问题。

    function Cat(name, age) {
      this.name = name;
      this.age = age;
      this.speak = function () {
        console.log('Hello I am ', this.name);
      }
    }
    
    let cat1 = new Cat('kiki1', 1);
    let cat2 = new Cat('kiki2', 2);
    
    cat1.speak(); //Hello I am  kiki1
    cat2.speak(); //Hello I am  kiki2
    
    // 检测对象类型
    console.log(cat1 instanceof Cat); // true
    console.log(cat2 instanceof Cat); // true
    
    // 不同实例上的同名函数是不相等
    console.log(cat1.speak == cat2.speak); // false
    

    特点:
    1、通过new Cat来创建实例
    2、可以通过instanceof来检测对象类型
    缺点:
    使用构造函数的缺点在于,每个方法都会在实例上重新创建一遍,所以不同实例上的同名函数是不相等的,即无法共享方法。比如上面的例子中的cat1.speak和cat2.speak是不相等的。

    为了解决此缺点,请看原型模式。

    六、原型模式

    function Cat() { }
    Cat.prototype.name = 'kiki1';
    Cat.prototype.age = 1;
    Cat.prototype.speak = function () {
      console.log('Hello I am ', this.name);
    }
    Cat.prototype.data = [1, 2];
    
    let cat1 = new Cat();
    let cat2 = new Cat();
    
    console.log(cat1.speak == cat2.speak); // true
    
    // 引用类型属性,不同实例之间互相影响
    console.log(cat2.data); // [1,2]
    cat1.data.push(3);
    console.log(cat1.data); // [1,2,3]
    console.log(cat2.data); // [1,2,3]
    

    特点:共享方法,比如上面的例子中,cat1.speak和cat2.speak是相等的。
    缺点:由于引用类型的属性也是共享的,因此不同实例之间会互相影响

    六、组合模式

    组合模式:即构造函数模式+原型模式。
    采用组合模式可以解决构造函数模式和原型模式的问题,又拥有构造函数和原型模式的特点,集两种模式之长。
    通过构造函数模式定义实例属性,通过原型模式定义共享方法和共享属性。

    function Cat(name, age, data) {
      this.name = name;
      this.age = age;
      this.data = data;
    }
    Cat.prototype.speak = function () {
      console.log('Hello I am ', this.name);
    }
    
    let cat1 = new Cat('kiki1', 1, [1, 2, 3]);
    let cat2 = new Cat('kiki2', 2, [3, 4, 5]);
    
    console.log(cat1.speak == cat2.speak); // true
    
    console.log(cat1.data); // [1,2,3]
    console.log(cat2.data); // [3,4,5]
    

    综上对比,组合模式是最优雅的创建对象的方式。

    相关文章

      网友评论

        本文标题:关于js(es5)如何优雅地创建对象

        本文链接:https://www.haomeiwen.com/subject/ltcefqtx.html