一、前颜(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 kiki1cat2.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 kiki1cat2.speak(); //Hello I am kiki2// 检测对象类型console.log(cat1 instanceof Cat); // trueconsole.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); // trueconsole.log(cat1.data); // [1,2,3]console.log(cat2.data); // [3,4,5]复制代码
综上对比,组合模式是最优雅的创建对象的方式。