js面向对象与原型
创建对象
var box = new Object();//创建对象
box.name = 'Lee'; //添加属性
box.age = 100;
box.run = function(){
return this.name + this.age + "运行中"; //this 表示当前作用域下对象
} // this 表示new Object()实例出来的那个对象
alert(box.run());
这就是创建对象最基本的方法,但是有个缺点,想创建一个类似的对象,就会产生大量的代码。
工厂模式
为了解决多个类似对象声明的问题,我们可以使用一种叫做工厂模式的方法,这种方法就是为了解决实例化对象产生大量重复的问题。
function createObject(name,age){
var obj = new Object();
obj.name = name;
obj.age = age;
obj.run = function(){
return this.name+this.age+"岁年龄";
}
return obj;
} var box1 = createObject('Lee',20);
var box2 = createObject('Jack',30);
console.log(box1.run());
console.log(box2.run());
工厂模式解决了重复实例化的问题,但还有一个问题,那就是识别问题,因为根本无法搞清他们到底是哪个对象的实例。
alert(typeof box1); //Object
alert(box1 instanceof Object);//true
构造函数
ECAMScript中采用构造函数(构造方法)可用来创建特定的对象。类似于Object对象。
//构造函数
function Box(name,age){
this.name = name;
this.age = age;
this.run = function(){
return this.name + this.age +"运行中...";
};
}; var box1 = new Box('Lee',100);
var box2 = new Box('Jack',200); console.log(box1.run());
console.log(box2.run());
使用构造函数的方法,即解决了重复实例化的问题,又解决了对象识别的问题,但问题是,这里并没有new Object(),为什么可以实例化Box(),这个是哪里来的呢?
使用了构造函数的方法,和使用工厂模式的方法他们不同之处如下:
- 1.构造函数方法没有显示的创建对象(new Objectt()),但它在后台自动var obj = new Object();
- 2.直接将属性和方法赋值给this对象,this就相当于obj;
- 3.没有return语句,不需要返回对象引用,它是在后台自动返回的。
//构造函数
function Box(name,age){
this.name = name;
this.age = age;
this.run = function(){
return this.name + this.age +"运行中...";
};
}; function Dack(name,age){
this.name = name;
this.age = age;
this.run = function(){
return this.name + this.age +"运行中...";
};
}; var box1 = new Box('Lee',100);
var box2 = new Box('Jack',200);
var box3 = new Dack('MrLee',300); console.log(box1.run());
console.log(box2.run());
console.log(box3.run()); //解决了对象识别问题
console.log(box1 instanceof Box); //true
console.log(box2 instanceof Box); //true
console.log(box3 instanceof Box); //false
console.log(box3 instanceof Dack);//true
对象冒充:使用call()方法
var o= new Object();
Box.call(o,'Lee',100);
console.log(o.run());
看下一个问题:
var box1 = new Box('Lee',100); //实例化后地址为1
var box2 = new Box('Lee',100); //实例化后地址为2 console.log(box1.name == box2.name); //true
console.log(box1.age == box2.age); //true
console.log(box1.run() == box2.run());//true //构造函数体内的方法的值是相当的
console.log(box1.run == box2.run); //false //因为他们比较的是引用地址
上面的代码运行说明引用地址不一样,那么构造函数内的方法也可以这样写:
this.run = new Function("return this.name + this.age +'运行'")
如何让他们的引用地址一样,下面代码:
function Box(name,age){
this.name = name;
this.age = age;
this.run = run;
}; function run(){
return this.name +this.age+"运行中...";
}
var box1 = new Box('Lee',100); //实例化后地址为1
var box2 = new Box('Lee',100); //实例化后地址为2
console.log(box1.run == box2.run); //true //因为他们比较的是引用地址
把构造函数内部的方法通过全局来实现引用地址一致。
虽然使用了全局函数run()来解决了保证引用地址一致的问题,但是这种方式又带来了一个新的问题,全局中的this在对象调用的时候是Box本身,而当普通函数调用的时候,this又代表window。
原型
function Box(){
//构造函数函数体内什么都没有,这里如有过,叫做实例属性,实例方法
} Box.prototype.name="Lee"; //原型属性
Box.prototype.age=100;
Box.prototype.run=function(){ //原型方法
return this.name+this.age+"运行中...";
} var box1=new Box();
var box2=new Box();
console.log(box1.run == box2.run); //true
console.log(box1.prototype);//这个属性是一个对象,访问不到
console.log(box1.__proto__);//这个属性是一个指针指向prototype原型对象。
如果是实例方法,不同的实例化,他们的方法地址是不一样的,是唯一的。
如果是原型方法,那么他们的地址是共享的,大家都一样。
console.log(box1.constructor); //构造属性,可以获取构造函数
PS:IE浏览器在脚本访问__proto__会不能识别,火狐和谷歌及其他某些浏览器能识别。虽然可以输出,但是无法获取内部信息。
判断一个对象是否指向该构造函数的原型对象,可以使用isPrototypeOf()方法来测试。
console.log(Box.prototype.isPrototypeOf(box1)); //true //只要实例化对象,即都会指向
原型模式的执行流程:
- 1.先查找构造函数实例里的属性或方法,如果有,立刻返回;
- 2.如果构造函数实例里没有,则去它的原型对象里找,如果有,就返回。
如何判断属性时构造函数的实例里,还是原型里?可以使用hasOwnProperty()函数来验证:
console.log(box1.hasOwnProperty('name'));//如果实例里有返回true,否则返回false
如何判断属性是原型里的?
function Box(){ } Box.prototype.name="Lee"; //原型属性
Box.prototype.age=100;
Box.prototype.run=function(){ //原型方法
return this.name+this.age+"运行中...";
} function isProperty(object,property){
return !object.hasOwnProperty(property) && (property in object);
}
var box1=new Box();
console.log(isProperty(box1,'name'));
为了让属性和方法更好的体现封装的效果,并且减少不必要的输入,原型的创建可以使用字面量的方式
使用字面量的方式创建原型对象,这里的{}就是对象,是object,new Object就相当于{}
function Box(){} Box.prototype={
name:'Lee',
age:100,
run:function(){
return this.name+this.age+"运行中...";
}
} var box = new Box();
console.log(box.constructor == Box); //false
字面量创建的方式使用constructor属性不会指向实例,而会指向Object,构造函数创建的方式则相反。
这里的Box.prototype={}就相当于创建了一个新的对象,所以 box.constructor是Object。
如何让box.constructor指向Box呢?
function Box(){} Box.prototype={
constructor:Box,//直接强制指向即可
name:'Lee',
age:100,
run:function(){
return this.name+this.age+"运行中...";
}
} var box = new Box();
console.log(box.constructor == Box); //true
重写原型,不会保留之前原型的任何信息,把原来的原型对象和构造函数对象的实例切断了。
function Box(){} Box.prototype={
constructor:Box,
name:'Lee',
age:100,
run:function(){
return this.name+this.age+"运行中...";
}
} //重写原型
Box.prototype={
age:200
}
var box = new Box();
console.log(box.name); //undefined
查看sort是否是Array原型对象里的方法
alert(Array.prototype.sort);
在如下 判断String原型对象里是否有substring方法
alert(String.prototype.substring);
给String 添加addstring方法:
String.prototype.addstring=function(){
return this+',被添加了!';
}
var box="Lee";
console.log(box.addstring());
注:原型模式创建对象也有自己的缺点,它省略了构造函数传参初始化这一过程,带来的缺点就是初始化的值都是一致的。而原型最大的缺点就是它最大的优点,那就是共享。
原型中所有属性是被很多实例共享的,共享对于函数非常合适,对于包含基本值的属性也还可以。但如果属性包含引用类型,就存在一定的问题:
function Box(){} Box.prototype={
constructor:Box,
name:'Lee',
age:100,
family:['哥哥','姐姐','妹妹'],
run:function(){
return this.name+this.age+"运行中...";
}
}; var box1 = new Box();
console.log(box1.family); //'哥哥','姐姐','妹妹'
box1.family.push("弟弟");
console.log(box1.family);//'哥哥','姐姐','妹妹','弟弟' var box2 = new Box();
console.log(box2.family);//'哥哥','姐姐','妹妹','弟弟'
从上面代码可以看出,在第一个实例修改后引用类型,保持了共享。box2.family共享了box1添加后的引用类型的原型。
为了解决构造传参和共享问题,可以组合构造函数+原型模式:
function Box(name,age){ //保持独立的用构造函数
this.name=name;
this.age=age;
this.family=['哥哥','姐姐','妹妹'];
} Box.prototype={ //保持共享的用原型
constructor:Box,
run:function(){
return this.name+this.age+"运行中...";
}
} var box1 = new Box('Lee',100);
console.log(box1.family); //'哥哥','姐姐','妹妹'
box1.family.push("弟弟");
console.log(box1.family);//'哥哥','姐姐','妹妹','弟弟' var box2 = new Box('Jack',200);
console.log(box2.family); //'哥哥','姐姐','妹妹' //引用类型没有使用原型,所以没有共享
动态原型模式
//把原型封装到构造函数里
function Box(name,age){
this.name=name;
this.age=age;
this.family=['哥哥','姐姐','妹妹']; console.log('原型初始化开始'); //执行了两次
Box.prototype.run=function(){
return this.name+this.age+"运行中...";
}
console.log('原型初始化结束'); //执行了两次
} //原型的初始化,只要第一次初始化就可以了,没必要每次构造函数实例化的时候都初始化
var box1 = new Box('Lee',100);
var box2 = new Box('Jack',200);
为了只让第一次初始化,那么就判断
function Box(name,age){
this.name=name;
this.age=age;
this.family=['哥哥','姐姐','妹妹']; if(typeof this.run!='function'){
console.log('原型初始化开始'); //执行了一次次
Box.prototype.run=function(){
return this.name+this.age+"运行中...";
};
console.log('原型初始化结束'); //执行了一次
}
} //原型的初始化,只要第一次初始化就可以了,没必要每次构造函数实例化的时候都初始化
var box1 = new Box('Lee',100);
var box2 = new Box('Jack',200);
寄生构造函数
如果以上都不能满足需要,可以使用一下寄生构造函数。
寄生构造函数=工厂模式+构造函数
function Box(name,age){
var obj = new Object();
obj.name=name;
obj.age=age;
obj.run=function(){
return this.name+this.age+"运行中...";
}
return obj;
} var box1 = new Box('Lee',100);
var box2 = new Box('Jack',200);
稳妥构造函数
在一些安全的环境中,比如禁止使用this和new,这里的this是构造函数里不使用的this,这里的new是在外部实例化构造函数时不使用new。这种创建方式叫做稳妥构造函数。
function Box(name,age){
var obj = new Object();
obj.name=name;
obj.age=age;
obj.run=function(){
return this.name+this.age+"运行中...";
}
return obj;
} var box1 = Box('Lee',100);
var box2 = Box('Jack',200);
继承
继承是面向对象中一个比较核心的概念。其它正统面向对象语言都会用两种方式实现继承:一个是接口实现,一个是继承。而ECMAScript只支持继承,不支持接口实现,而实现继承的方式依靠原型链完成。
function Box(){
this.name="Lee";
} function Jack(){
this.age=100;
} Jack.prototype = new Box(); var jack = new Jack();
console.log(jack.name); //Lee
为了解决引用共享和超类型无法传参的问题,我们采用一种叫借用构造函数的技术,或者成为对象冒充(伪造对象、经典继承)的技术解决这两个问题。
function Box(name){
this.name=name;
} Box.prototype.age=200; function Jack(name){
Box.call(this,name);
} var jack = new Jack('Lee'); console.log(jack.name);//Lee
console.log(jack.age);//undefined
但是上面的代码可以看出,对象冒充没有继承原型链上的age属性。所以要继承Box的原型,就出现下面的组合继承。
组合继承即是原型链+借用构造函数的模式
function Box(name){
this.name=name;
} Box.prototype.age=200; function Jack(name){
Box.call(this,name);
} Jack.prototype = new Box(); var jack = new Jack('Lee'); console.log(jack.name);//Lee
console.log(jack.age);//
原型式继承
//临时中转函数
function obj(o){
function F(){};
F.prototype = o;
return new F();
} //这是字面量的声明方式,相当于var box = new Box();
var box={
name:'Lee',
age:100,
family:['哥哥','姐姐','妹妹']
}; var box1 = obj(box);
console.log(box1.family);//'哥哥','姐姐','妹妹'
box1.family.push('弟弟');
console.log(box1.family);//'哥哥','姐姐','妹妹','弟弟' var box2 = obj(box);
console.log(box2.family);//'哥哥','姐姐','妹妹','弟弟'
存在的问题就是引用类型共享了。
寄生式继承
把原型式与工厂模式结合起来。
//临时中转函数
function obj(o){
function F(){};
F.prototype = o;
return new F();
} //寄生函数
function create(o){
var f=obj(o);
f.run=function(){
return this.name+"方法";
}
return f;
} //这是字面量的声明方式,相当于var box = new Box();
var box={
name:'Lee',
age:100,
family:['哥哥','姐姐','妹妹']
}; var box1 = create(box);
console.log(box1.run());
寄生组合继承
//临时中转函数
function obj(o){
function F(){};
F.prototype = o;
return new F();
} //寄生函数
function create(box,desk){
var f=obj(box.prototype);
f.constructor=desk; //调整原型构造指针
desk.prototype=f;
} function Box(name,age){
this.name=name;
this.age=age;
} Box.prototype.run=function(){
return this.name+this.age+"运行中...";
} function Desk(name,age){
Box.call(this,name,age); //对象冒充
} //通过寄生组合继承来实现继承
create(Box,Desk); //这句话用来替代Desk.prototype = new Box(); var desk = new Desk('Lee',100);
console.log(desk.run());
js面向对象与原型的更多相关文章
- JS面向对象之原型
面向对象之原型 为什么要使用原型 由于 js 是解释执行的语言, 那么在代码中出现的函数与对象, 如果重复执行, 那么会创建多个副本, 消耗更多的内存, 从而降低性能 传统构造函数的问题 functi ...
- JS面向对象,原型,继承
ECMAScript有两种开发模式:1.函数式(过程化),2.面向对象(OOP).面向对象的语言有一个标志,那就是类的概念,而通过类可以创建任意多个具有相同属性和方法的对象.但是,ECMAScript ...
- JS面向对象之原型链
对象的原型链 只要是对象就有原型 原型也是对象 只要是对象就有原型, 并且原型也是对象, 因此只要定义了一个对象, 那么就可以找到他的原型, 如此反复, 就可以构成一个对象的序列, 这个结构就被成 ...
- 从零开始的全栈工程师——JS面向对象( 原型 this 继承)
一.生成对象的方式 ①单例模式(字面量定义)var obj = {} ②类的实例 var obj = new Object( ) ③工厂模式 ④构造函数:扮演三种角色 普通函数 普通对象 类 工厂模式 ...
- JS面向对象——动态原型模型、寄生构造模型
动态原型模型 组合使用构造函数模型和原型模型,使得OO语言程序员在看到独立的构造函数和原型时很困惑.动态原型模型致力于解决该问题,它把所有的信息封装在构造函数中,通过在构造函数中初始化原型(仅在必要情 ...
- JS面向对象(2) -- this的使用,对象之间的赋值,for...in语句,delete使用,成员方法,json对象的使用,prototype的使用,原型继承与原型链
相关链接: JS面向对象(1) -- 简介,入门,系统常用类,自定义类,constructor,typeof,instanceof,对象在内存中的表现形式 JS面向对象(2) -- this的使用,对 ...
- js面向对象-原型链
var Person = function (name) { this.name = name; } Person.prototype.say = function () { console.log( ...
- 【JavaScript】 JS面向对象的模式与实践 (重点整治原型这个熊孩子 (/= _ =)/~┴┴ )
参考书籍 <JavaScript高级语言程序设计>—— Nicholas C.Zakas <你不知道的JavaScript> —— KYLE SIMPSON 在JS的面向 ...
- js面向对象、创建对象的工厂模式、构造函数模式、原型链模式
JS面向对象编程(转载) 什么是面向对象编程(OOP)?用对象的思想去写代码,就是面向对象编程. 面向对象编程的特点 抽象:抓住核心问题 封装:只能通过对象来访问方法 继承:从已有对象上继承出新的对象 ...
随机推荐
- 从零开始学 Java - Windows 下安装 JDK
关于未来 "我要死在火星.在我死去的时候能够想着人类能有一个美好的未来--有可持续的能源,同时能够殖民其他的星球来避免人类灭绝的最坏可能." 官网下载 直接打开官网:http:// ...
- 【http抓包】记录一次抓手机app的接口
抓手机的接口地址,好用的工具很多,想 windows下的 Fiddler 和mac下的Charles 1. fiddler的设置教程是 http://jingyan.baidu.com/article ...
- JAVA中常说的三大框架指:SSH
即:spring.Struts.hibernate Spring:功能强大的组件粘合济,能够将你的所有的Java功能模块用配置文件的方式组合起来(还让你感觉不到spring的存在)成为一个完成的应用 ...
- JavaScript闭包理解【关键字:普通函数、闭包、解决获取元素标签索引】
以前总觉得闭包很抽象,很难理解,所以百度一下"闭包"概览,百度的解释是:“闭包是指可以包含自由(未绑定到特定对象)变量的代码块:这些变量不是在这个代码块内或者任何全局上下文中定义的 ...
- HTML5 学习笔记(四)——canvas绘图、WebGL、SVG
一.Canvas canvas是HTML5中新增一个HTML5标签与操作canvas的javascript API,它可以实现在网页中完成动态的2D与3D图像技术.<canvas> 标记和 ...
- Senna.js – 速度极快的单页应用程序引擎
Senna.js 是一个速度超快的单页应用程序引擎,提供了几个低级别的 API,可以帮助你打造现代化的基于 Web 的应用程序.更重要的是,搜索引擎蜘蛛应该能够索引相同的内容. 通过使用 HTML5 ...
- jquery定时滑出可最小化的底部提示层
效果预览:http://keleyi.com/keleyi/phtml/jqtexiao/index.htm当打开页面或者刷新页面后等待两秒钟,会在底部滑出可最小化的提示层.滑出层半透明,可关闭再现. ...
- js实现web网页版台球游戏
js桌球小游戏在线试玩地址:http://keleyi.com/game/13/ 游戏截图: 完整代码,保存到html文件可以试玩: <!DOCTYPE html PUBLIC "-/ ...
- Jquery中的Ajax
AJAX: * jQuery中的Ajax * 封装第一层 - 类似于原生Ajax的用法 * $.ajax() - 最复杂 * 选项 * url - 请求地址 * type - 请求类型,默认为GET ...
- 认识DOM和一些方法
认识DOM 文档对象模型DOM(Document Object Model)定义访问和处理HTML文档的标准方法.DOM 将HTML文档呈现为带有元素.属性和文本的树结构(节点树). 先来看看下面代码 ...