创建对象

js高程3--第6章面向对象的程序设计--第二节创建对象,批量创建对象有很多种模式,每一种模式都有自己的优点与缺点,搞清楚它们出现的历史原因,优缺点,我们才能使用的更加游刃有余!

本片文章并没有将细节的挖的特别深,重点关注的是每种模式的优缺点,怎么形成的。细节会另写博客总结。

正文开始

我们都知道创建单个对象有两种方法,构造函数字面量的形式。

var obj = new Object(); //构造函数
var obj1 = {}; //字面量

如果我们想要创建多个对象,这两种形式就有明显的缺点:每一个新对象都要手写生成,会产生大量重复的代码!

为了解决这个问题,聪明的程序员开始使用工厂模式

工厂模式

用函数来封装以特定接口创建对象的细节

function createPerson(name,age){
var obj = new Object();
obj.name = name;
obj.age = age;
obj.sayName = function(){
alert(this.name);
}
return obj;
}
var zhangsan = creatPerson('zhangsan',18);

优点:解决了创建多个相似对象,代码重复问题

缺点:没有解决对象识别问题,即不知道当前对象的类型,一直都是Object类型

随着javascript的发展,又一个新的模式出现了!

构造函数模式

ECMAScript中的构造函数可以用来创建特定类型的对象。

类似Object,Array这种原生的构造函数,我们可以使用new Object(),new Array()来创建对象类型或者数组类型的对象。

同样我们可以构建自定义类型的构造函数,定义自定义类型的属性和方法。

function Person(name,age){
this.name = name;
this.age = age;
this.sayName = function(){
alert(this.name);
}
} var zhangsan = new Person('zhangsan',18);

优点:可以使用instanceof,isPrototypeOf用来识别对象的类型

缺点 :每个方法都要在每个实例上重新创建一遍,浪费内存。

有人觉得既然实例的sayName函数的作用相同,就没有必要提前把函数绑定到构造函数中,可以这样

function Person(name,age){
this.name = name;
this.age = age;
this.sayName = sayName;
}
function sayName(){ //变为全局函数
alert(this.name);
}
var zhangsan = new Person('zhangsan',18);

看似很好的解决了每次都重新创建的问题,但是这样又延伸出一个新的让人无法接受的问题,全局变量污染,这样根本没有封装性可言。

very lucky!这些问题都可以通过原型模式解决。

原型模式

function Person(){}

Person.prototype.name = 'zhang';
Person.prototype.age = 18;
Person.prototype.sayName = function(){
alert(this.name);
}

这样每一个实例的__proto__属性引用Person的原型属性地址,共用一个原型对象。

但是上面的写法每次添加属性都要Person.prorotype很麻烦,可以使用下面字面量的形式

function Person(){}

Person.prototype = {
name:'zhang',
age:18,
sayName:function(){
alert(this.name);
}
} var zhang = new Person();

但是字面量的写法会切断原型链,尽管这里使用instanceofisPrototypeOf仍然可以判断实例的类型,

但是我们通过zhang.constructor得到的是Object,而不再是Person了,因为我们改变了Person的原型属性默认的地址,新地址constructor属性是不存在的。

所以为了是原型链完整,可以使用下面的形式

function Person(){}

// 也可在这里直接添加constructor,
// 只不过constructor属性特征跟默认的(不可枚举)是不一样的
Person.prototype = {
name:'zhang',
age:18,
sayName:function(){
alert(this.name);
}
} // 手动添加构造函数属性,保证原型链完整,
Object.defineProperty(Person.prototype,'constructor',{
value:Person,
enumerable:false,
}) var zhang = new Person();

但是,这样还是会有问题,所有的实例都是一样的属性和方法,没有私人属性和方法,这当然不是我们想要的。

解决这个问题很简单,把构造函数模式原型模式混合使用

function Person(name,age){
this.name = name;
this.age = age;
} Person.prototype = {
sayName:function(){
alert(this.name);
}
}
var zhang = new Person(); // 手动添加构造函数属性,保证原型链完整,
Object.defineProperty(Person.prototype,'constructor',{
value:Person,
enumerable:false,
})

Perfect!!!这种组合构造函数和原型模式是目前使用最为广泛的模式,可以说是创建自定义类型的默认模式了。

但是,-_-,哈哈 心中一万只神兽,竟然还有但是!!!

严格面向对象的语言使用者看见这种形式会非常懵逼,说好的封装,竟然不把所有信息封装在构造函数里面,竟然还有独立的原型,wtf!

程序员嘛,多少会有点的强迫症。

动态原型模式

这个模式非常的巧妙,第一次看见非常的兴奋。

function Person(name,age){
this.name = name;
this.age = age;
// 第一次初始化构造函数的时候执行,之后不再执行
// 这里的属性判断随便一个原型属性都可以
if(typeof this.sayName != 'function'){
Person.prototype = {
sayName:function(){
alert(this.name);
}
}
}
} var zhang = new Person();

当得了当~~~,如果不加这个判断,每次实例化都执行一遍原型是不现实的,加个判断完美解决了这个问题,这种解决问题的思想是值得非常借鉴的,可以应用到我们的实际开发中。

寄生构造函数模式

还没有搞明白,待续

稳妥构造函数模式

还没有搞明白,待续

补充

要真正搞清楚对象这一块,需要清楚一下几个知识点:

Object

instanceof

getPrototypeOf

isPrototypeOf

hasOwnProperty

defineProperty

in

本人能力有限,理解的不对的欢迎指正!

js高程3--面向对象的程序设计--创建对象的更多相关文章

  1. 重学js之JavaScript 面向对象的程序设计(创建对象)

    注意: 本文章为 <重学js之JavaScript高级程序设计>系列第五章[JavaScript引用类型]. 关于<重学js之JavaScript高级程序设计>是重新回顾js基 ...

  2. JS_高程6.面向对象的程序设计(2)创建对象_3 构造函数存在的问题

    # 上次讲到用构造函数的模式来创建对象,相对于工厂模式,解决可对象识别的问题. function Person(name,age,job){ this.name=name; this.age=age; ...

  3. JS_高程6.面向对象的程序设计(2)创建对象_1

    一.创建对象的常见方法 (1)Object构造函数创建单个对象,早期的JavaScript开发人员经常使用该模式创建新对象. var person=new Object(); person.name= ...

  4. JS_高程6.面向对象的程序设计(2)创建对象_2 构造函数也是一般函数

    1.构造函数也是一般函数,以下创建一个构造函数. var Person=function(name,age,job){ this.name=name; this.age=age; this.job=j ...

  5. JS_高程6.面向对象的程序设计(1)理解对象

    js的数据属性:P139(1)[[Configurable]](2)[[Enumerable]](3)[[Writable]](4)[[Value]] 使用Object.definerPropert( ...

  6. 《JS高程》创建对象的7种方式(完整版)

    一.理解对象 ECMA-262定义对象:无序属性的集合,其属性可以包含基本值.对象或者属性. 我们可以把 ECMAScript 的对象想象成 散列表:无非就是一组 名值对,其中值可以是数据或函数. 创 ...

  7. 吃透Javascript数组操作的正确姿势—再读《Js高程》

    Javascript中关于数组对象的操作方法比较多也比较杂,正好再次捡起<Javascript高级程序设计>来读,把它们一一总结梳理了一下: 方法类别 方法名称 方法描述 参数 返回值 备 ...

  8. 《JS高程》JS-Object对象整理

    继上篇<JS高程>对象&原型笔记,对JavaScript中的Object对象进行了整理,梳理一遍~ 参考文章:详解Javascript中的Object对象 ------------ ...

  9. Day046--JavaScript-- DOM操作, js中的面向对象, 定时

    一. DOM的操作(创建,追加,删除) parentNode 获取父级标签 nextElementSibling 获取下一个兄弟节点 children 获取所有的子标签 <!DOCTYPEhtm ...

随机推荐

  1. Android Native 内存泄漏系统化解决方案

    导读:C++内存泄漏问题的分析.定位一直是Android平台上困扰开发人员的难题.因为地图渲染.导航等核心功能对性能要求很高,高德地图APP中存在大量的C++代码.解决这个问题对于产品质量尤为重要和关 ...

  2. VC win32 static library静态链接库简单示例

    中午在宿舍闲来没事,看到网上一篇帖子,关于静态链接库的英文示例.它在.Net上开发,我将其移到VC上开发,因此对其代码做了相应修改.帖子内容如下:(代码我已修改).原帖见:http://msdn.mi ...

  3. 一文带你了解git

    git简介 什么是git? git是当今世界上最先进的分布式的版本控制系统. 版本控制系统分集中式的和分布式的,集中式的主要代表有CVS.SVN,而Git是分布式版本控制系统的佼佼者. 那什么是集中式 ...

  4. 2019年7月20日 - LeetCode0003

    https://leetcode-cn.com/problems/longest-substring-without-repeating-characters/submissions/ 我的解法: c ...

  5. Spring管理事物两种方式

    Spring管理事物两种方式 1. 编程式事物管理(在开发中不经常使用) 使用步骤 1. 配置数据库事物管理 DataSourceTransactionManager <!--配置事物管理器-- ...

  6. sql LocalDB 的安装环境和使用方法

    LocalDB LocalDB专门为开发商.它是非常容易安装,无需管理,但它提供了相同的T-SQL语言,编程表面和客户端供应商定期的SQL Server Express.实际上,目标SQL Serve ...

  7. spark 源码分析之十--Spark RPC剖析之TransportResponseHandler、TransportRequestHandler和TransportChannelHandler剖析

    spark 源码分析之十--Spark RPC剖析之TransportResponseHandler.TransportRequestHandler和TransportChannelHandler剖析 ...

  8. 高级查询MYsql(二) 练习

    一.单词部分 ①exist存在②temp临时的③district区域 ④content内容⑤temporary暂时的 二.预习部分 1.表连接都可以用子查询替换吗 是的 2.检测某列是否存在某个范围可 ...

  9. jmeter Linux环境执行总报错 cannot allocate memory

    1.windows环境写好的测试用例,执行没有问题,在Linux环境跑总是报错,提示如下 cannot allocate memory 2.一开始以为是哪块设置有问题,因为脚本里边有设置邮件自动发送, ...

  10. isMemberOfClass、isKindOfClass原理分析

    isMemberOfClass - 调用者必须是传入的类的实例对象才返回YES- 判断调用者是否是传入对象的实例,别弄反了,如 [s1 isMemberOfClass:p1] ,意思是s1是否是p1的 ...