Javascript学习笔记:9种创建对象的方式
最基本的对象创建方式是通过Object构造函数或对象字面量的方式创建:
①通过Object构造函数的方式创建对象:
var person=new Object();//或者写成var person={}
person.name='张三';
②通过对象字面量的方式创建对象:
var person={name:'张三'};
Object构造函数模式和对象字面量模式这两种创建对象的方式,都有明显的缺点:使用同一个接口创建很多对象的时候,会产生大量的重复代码。为解决这个问题,可以使用工厂模式。工厂模式就是以函数的方式来封装以特定接口创建对象的细节。
③以工厂模式的方式创建对象:
function createPerson(name,sex){
var person={};
person.name=name;
person.sex=sex;
return person;
}
var person=createPerson('张三','男');
工厂模式创建对象的方式虽然解决了重复代码的问题,但是还有一个重要的问题还有待解决,那就是对象识别的问题,就是所有以工厂模式创建对象的方式产生的对象都只能判断是Object类型,不能够进一步识别该对象。为了解决这一问题,我们可以使用构造函数模式。
④以构造函数模式的方式创建对象:
function Person(name,sex){
this.name=name;
this.sex=sex;
}
var p=new Person('张三','男');
以构造函数模式创建对象的方式与以工厂模式创建对象的方式的不同之处在于:a、没有显示的创建对象;b、直接将属性和方法付给了this;c、没有return语句。同时要创建Person的实例,必须用new操作符。通过构造函数的方式创建的对象可以通过p.constructor==Person或者p instanceof Person来判定对象的类型。虽然以构造函数的方式创建对象解决了对象识别的问题,但是仍然有一些问题,就是该引用类型存在方法属性时,每个方法都要在每个实例上重新创建一遍。我们可以使用原型模式来解决这个问题。原型模式就是在构造函数的prototype上添加属性和方法,这些方法和属性是所有通过该构造函数生成的对象实例共享的。使用原型模式的好处就是可以让所有对象实例共享原型对象上的属性和方法。
⑤以原型模式的方式创建对象:
function Person(){
}
Person.prototype.name='张三';
Object.defineProperty(Person.prototype,'sex',{configurable:true,writable:true});
var p=new Person();
以原型模式创建对象的最大缺点是,所有的属性都是所有实例共享的,因此任何一个实例的属性更改了,那么其他实例的该属性也会更改。为了解决这个问题,可以组合使用构造函数模式和原型模式。
⑥以组合使用构造函数模式和原型模式的方式创建对象:
function Person(name,age){
this.name=name;
this.age=age;
}
Object.defineProperty(Person.prototype,'birth',{get:function(){
return ((new Date()).getFullYear()-this.age);
}})
var p=new Person('张三',25);
console.log(p.birth); //
一般的,把需要共享的属性和方法放在原型中,把不需要共享的属性放在构造函数中。
使用组合使用构造函数模式和原型模式的方式,构造函数和原型的代码是独立分开的,这会让人感觉很困惑,为了让两者放在一起,我们可以使用动态原型模式。
⑦以动态原型模式创建对象:
function Person(name,age){
this.name=name;
this.age=age;
if(typeof getBirth !=='function'){
Person.prototype.getBirth=function(){
return ((new Date()).getFullYear()-this.age);
}
}
}
var p=new Person('张三',25);
原型对象上的方法只会在生成第一个实例的时候添加到原型对象上。
对于已有的引用类型,不能给该引用类型添加属性(如果直接添加在该引用类型的原型对象上,那么会影响所有的该引用对象生成的实例),比如说希望创建一个具有特殊方法的数组,由于不能直接修改Array构造函数,也不能给Array的原型对象上添加方法,我们可以使用寄生构造函数模式。寄生构造函数模式就是拥有构造函数的外表,构造函数内部用的是工厂模式的方式生成新的实例。
⑧以寄生构造函数的方式创建对象:
function SpecialArray(){
var values=new Array();
values.push.apply(values,arguments);
//添加特殊方法
values.toPipedString=function(){
return this.join("|");
}
return values;
}
var sa=new SpecialArray("张三","李四","王五");
console.log(sa.toPipedString()); //张三|李四|王五
当我们希望我们创建的对象拥有私有变量,只有该对象的公有方法可以修改私有变量的值,我们可以使用稳妥构造函数模式。稳妥构造函数模式类似寄生构造函数模式,只是不将属性直接设置在对象实例上,而是通过闭包的方式,为对象实例添加可以访问私有属性的方法。
⑨以稳妥构造函数模式的方式创建对象:
function Person(name){
var o =new Object();
o.getName=function(){
return name;
};
return o;
}
var p=Person("张三");
console.log(p.getName());//张三
console.log(p.name);//undefined
稳妥构造函数主要用于一些安全环境中(这些环境中会禁止使用new和this),或者用于防止数据被其他应用程序改动时使用。注意这里的引用类型的公有方法使用不能使用this.来取得私有属性。
Javascript学习笔记:9种创建对象的方式的更多相关文章
- JavaScript:学习笔记(7)——VAR、LET、CONST三种变量声明的区别
JavaScript:学习笔记(7)——VAR.LET.CONST三种变量声明的区别 ES2015(ES6)带来了许多闪亮的新功能,自2017年以来,许多JavaScript开发人员已经熟悉并开始使用 ...
- Java程序猿的JavaScript学习笔记(8——jQuery选择器)
计划按例如以下顺序完毕这篇笔记: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScript ...
- Java程序猿JavaScript学习笔记(2——复制和继承财产)
计划和完成在这个例子中,音符的以下序列: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaSc ...
- Java程序猿的JavaScript学习笔记(3——this/call/apply)
计划按例如以下顺序完毕这篇笔记: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScript ...
- Javascript学习笔记-一些关键点
Javascript学习笔记-一些关键点 Table of Contents 1. 调试 2. == vs === 3. 两种函数声明 4. 技术感悟 1 调试 现在的主流浏览器都提供了开发者模式,可 ...
- Javascript学习笔记四——操作表单
Javascript学习笔记 大多网页比如腾讯,百度云之类的需要登陆,用户输入账号密码就可以登陆,那么浏览器是如何获取用户的输入的呢?今天就记录一下操作表单. 操作表单与操作DOM是差不多的,表单本身 ...
- .NET Remoting学习笔记(二)激活方式
目录 .NET Remoting学习笔记(一)概念 .NET Remoting学习笔记(二)激活方式 .NET Remoting学习笔记(三)信道 参考:百度百科 ♂风车车.Net 激活方式概念 在 ...
- Java程序猿JavaScript学习笔记(4——关闭/getter/setter)
计划和完成这个例子中,音符的顺序如下: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScr ...
- Java程序猿的JavaScript学习笔记(1——理念)
计划按例如以下顺序完毕这篇笔记: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScript ...
随机推荐
- Unity学习疑问记录之时间变量
1.Time.deltaTime 以秒计算,完成最后一帧的时间 放在Update()函数中的代码是以帧来执行的.如果我们需要物体的移动以秒来执行.我们需要将物体移动的值乘以Time.deltaTime ...
- Codeforces Round #375 (Div. 2) A B C D F
A 数轴上有三个人要到一个点去过年 使三个人走路距离的和最小 让两边的人都走到中间那个点即可 B 给出一个字符串 其中有_ ( ) 三种字符和英文字母 连续的英文字母是一个单词 括号对中不会套括号对 ...
- Socket请求和Http请求的各自特点、区别及适用场景
Socket实现服务器与客户端之间的物理连接,并进行数据传输.主要有TCP/UDP两个协议.Socket处于网络协议的传输层.TCP:传输控制协议,面向连接的的协议,稳定可靠.当客户和服务器彼此交换数 ...
- php 生成唯一id的几种解决方法
php 生成唯一id的几种解决方法 网上查了下,有很多的方法 1.md5(time() . mt_rand(1,1000000)); 这种方法有一定的概率会出现重复 2.php内置函数uniqid ...
- ExtJS笔记 Form
A Form Panel is nothing more than a basic Panel with form handling abilities added. Form Panels can ...
- XHTML基础
简介:前一章,我们知道网页主要是由内容.结构.表现和行为四个部分组成,而网页的结构由W3C规定的XHTML语言定义.本章介绍定义网页结构的XHTML基本标价. 1.XHTML基本语法 ...
- 程序中的@Override是什么意思
@Override是Java5的元数据,自动加上去的一个标志,告诉你说下面这个方法是从父类/接口 继承过来的,需要你重写一次,这样就可以方便你阅读,也不怕会忘记 @Override是伪代码,表示重写( ...
- 为ASP.NET MVC视图输出json
做个小小练习,为asp.net mvc视图输出json字符串: 创建JsonResult操作: 创建此视图: 浏览结果:
- 通过 Chrome Workspace 调试本地项目(修改样式时及时保存)
打开 DevTools 开发者工具[F12]中的 Sources 面板,在面板左侧右键选择「Add folder to workspace」,选择添加的文件夹. 添加好后,右键一个文件,选择「Map ...
- Elasticsearch学习笔记(一)
批量建索引: curl -s -XPOST 'localhost:9200/_bulk' --data-binary @documents.json 查看索引mappingmyindex/_mappi ...