js高程3--面向对象的程序设计--创建对象
创建对象
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();
但是字面量的写法会切断原型链,尽管这里使用instanceof,isPrototypeOf仍然可以判断实例的类型,
但是我们通过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--面向对象的程序设计--创建对象的更多相关文章
- 重学js之JavaScript 面向对象的程序设计(创建对象)
注意: 本文章为 <重学js之JavaScript高级程序设计>系列第五章[JavaScript引用类型]. 关于<重学js之JavaScript高级程序设计>是重新回顾js基 ...
- JS_高程6.面向对象的程序设计(2)创建对象_3 构造函数存在的问题
# 上次讲到用构造函数的模式来创建对象,相对于工厂模式,解决可对象识别的问题. function Person(name,age,job){ this.name=name; this.age=age; ...
- JS_高程6.面向对象的程序设计(2)创建对象_1
一.创建对象的常见方法 (1)Object构造函数创建单个对象,早期的JavaScript开发人员经常使用该模式创建新对象. var person=new Object(); person.name= ...
- JS_高程6.面向对象的程序设计(2)创建对象_2 构造函数也是一般函数
1.构造函数也是一般函数,以下创建一个构造函数. var Person=function(name,age,job){ this.name=name; this.age=age; this.job=j ...
- JS_高程6.面向对象的程序设计(1)理解对象
js的数据属性:P139(1)[[Configurable]](2)[[Enumerable]](3)[[Writable]](4)[[Value]] 使用Object.definerPropert( ...
- 《JS高程》创建对象的7种方式(完整版)
一.理解对象 ECMA-262定义对象:无序属性的集合,其属性可以包含基本值.对象或者属性. 我们可以把 ECMAScript 的对象想象成 散列表:无非就是一组 名值对,其中值可以是数据或函数. 创 ...
- 吃透Javascript数组操作的正确姿势—再读《Js高程》
Javascript中关于数组对象的操作方法比较多也比较杂,正好再次捡起<Javascript高级程序设计>来读,把它们一一总结梳理了一下: 方法类别 方法名称 方法描述 参数 返回值 备 ...
- 《JS高程》JS-Object对象整理
继上篇<JS高程>对象&原型笔记,对JavaScript中的Object对象进行了整理,梳理一遍~ 参考文章:详解Javascript中的Object对象 ------------ ...
- Day046--JavaScript-- DOM操作, js中的面向对象, 定时
一. DOM的操作(创建,追加,删除) parentNode 获取父级标签 nextElementSibling 获取下一个兄弟节点 children 获取所有的子标签 <!DOCTYPEhtm ...
随机推荐
- Bzoj1972: [Sdoi2010]猪国杀 题解(大模拟+耐心+细心)
猪国杀 - 可读版本 https://mubu.com/doc/2707815814591da4 题目可真长,读题都要一个小时. 这道题很多人都说不可做,耗时间,代码量大,于是,本着不做死就不会死的精 ...
- 「玩转Python」突破封锁继续爬取百万妹子图
前言 从零学 Python 案例,自从提交第一个妹子图版本引来了不少小伙伴的兴趣.最近,很多小伙伴发来私信说,妹子图不能爬了!? 趁着周末试了一把,果然爬不动了,爬下来的都是些 0kb 的假图片,然后 ...
- 缓存实践Cache Aside Pattern
Cache Aside Pattern旁路缓存,是对缓存应用的一个总结,包括读数据方案和写数据方案. 读数据方案 先读cache,如果命中则返回 如果miss则读db 将db的数据存入缓存 写数据方案 ...
- ZIP:ZipFile
ZipFile: /* 此类用于从 ZIP 文件读取条目 */ ZipFile(File file) :打开供阅读的 ZIP 文件,由指定的 File 对象给出. ZipFile(File file, ...
- Excel催化剂开源第16波-VSTO开发之脱离传统COM交互以提升性能
在VSTO开发或其他COM技术开发过程中,甚至VBA也是,在和Excel交互中,难免会遇到性能瓶颈问题,COM技术的交互实在太慢,对大量数据读写等操作,耗时太长,容易卡用户界面以为是程序死机等等. 在 ...
- 小白学python-day05-IDE、格式化输出、For While循环、断点、continue、break
今天是day05,以下是学习总结. 但行努力,莫问前程. ----------------------------------------------------------------------- ...
- Codeforces Round #486 (Div. 3) C "Equal Sums" (map+pair<>)
传送门 •题意 给k个数列,从中k个数列中找出任意2个数列 i ,j 使得数列i删除第x个数,和数列j删除第y个数的和相等 若存在,输出 i ,x 和 j,y •思路 每个数列之间的联系为数列的和之间 ...
- Oracle 数据库登录、用户解锁、改密码、创建用户授权操作
一.数据库登录1.常用账户: 管理员: sys主要练习操作用户: scott2.测试环境是否配置成功: 1.命令窗口 win+R -> cmd(以管理员身份运行) - > sqlplus ...
- java中代码的注释和快捷
添加必要的注释,对一个有责任心.有道德模范的前端必须具备的好习惯, 可以大大提高代码的可维护性.可读性. java代码注释快捷键:ctrl+shift+/首先熟悉一下html.css.js的注释的写法 ...
- 计数排序and基数排序
1 计数排序,稳定 复杂度o(k + n) public static int[] countingSort(int[] nums) { int n = nums.length; ; ; i & ...