OOP in Javascript
写了几篇Vue入门的内容了,今天写点其它的放松一下,简单讲讲javascript中的面相对象。
在面向对象的语言中,都有类的概念,当然es6中开始javascript中也有类的概念了,这里以es5为基础开始讲解,毕竟当前即使写的es6代码,一般还是会通过babel等转码器翻译到es5来执行的;
在js中虽然没有类的概念,但是我们却可以创建对象,一般创建对象有两种方式(这里指自定义对象):
1、使用构造函数
function Person(){
}
var p=new Person();
2、使用字面量
var p={
//各种属性
}
这里着重介绍一下构造函数方式创建对象;以上面代码为例,function Person(){},这个和普通函数有什么区别么?其实除了命名约定外(建议首字母大写)声明上没有任何区别,主要是调用方式的不同,构造函数调用使用new 操作符,使用new 操作符调用构造函数,主要经过四个步骤:
1)、创建一个新的对象;
2)、把this指向新对象;
3)、给对象添加属性
4)、返回新对象
对我们上面的代码进行一点简单的修改:
function Person(){
this.name= "zhangsan";
this.age=10;
};
var p=new Person();
console.log(p.name,p.age) ;//'zhangsan 10'
再对比上面使用new操作符调用构造函数的过程,是不是很容易理解输出的结果。
刚才也说了,构造函数和普通函数调用完全相同,那么是不是说可以把构造函数进行普通调用呢?当然可以,对我们的例子再次进行修改:
function Person(){
this.name= "zhangsan";
this.age=10;
};
//var p=Person();
Person();
console.log(window.name,window.age) ;//'zhangsan 10'
不使用new操作符调用,不会经过上述四个步骤的隐式处理,所以这个时候不会有新对象的创建和this指向的改变,那么此时的this就指向了全局对象,在浏览器中即window对象,所以可以使用window.name来访问,得到正确的输出。
OOP三大特性:封装、继承、多态;这里只讲述在javascript中如何进行继承(限es5)。
继承只是是代码重用的一种手段,那么在js中如何实现继承呢?
基于原型的继承
首先,函数也是对象,因为所有函数都是Function对象的实例,function 函数名,这种方式定义函数只是一种快捷方式,理论上和var 函数名= new Function(p1,body)等价,从这一点上看函数名其实就是指向函数的指针,但是两者不同的是,通过new Function来创建函数,会被js解释器解释两次,一次是声明时,第二次是对body部分的解析。
其次,任何一个函数创建后,都有一个原型对象,如:
声明一个Person函数,那么我们可以使用Person.prototype来打印一下其原型对象,可以看到其原型对象是一个Object类型,里面包括一个属性(不算__proto__,这个术语内部变量,是实例到原型的指针)constructor,指向了函数Person;另外原型对象本身就是另一个对象的实例(这是是Object函数的实例,=new Object()),任何一个对象的实例,都包含一个内部变量__proto__(chrome浏览器)(Object.create(null)创建的对象除外),指向创建这个对象实例的类型的原型,此处__prop__指向的是Object.prototype.而且可以看到Object.prototype上包含的几个方法(toString、valueOf等)。
上面说了任何一个对象的实例,都包含一个指向其构造函数原型的内部变量,那么我们创建一个Person 的实例:
可以看到是同一个,所以我们可以总结构造函数、实例、和构造函数原型之间的关系如下:
虽然画的是很难看,但是我觉得应该也表达出我要表达的意思了啊,?。
综上,构造函数的原型是一个对象,默认情况下,该对象是Object对象的实例;由于访问时,查找规则如下:先查找当前对象的实例属性,如果找到则返回,否则,查找__proto__指向的原型对象上的同名属性,找到则返回,否则继续原型对象的__proto__指向的对象上的同名属性,一直到Object 实例的__proto__,也就是Object.prototype为止,上述由__proto__构建的这个链接就叫做原型链。
请看如下代码:
function Person(){
this.name="zhangsan";
this.myFriends=["zhangsan","lisi"]
}
Person.prototype.getName=function(){
return this.name;
}
function Student(){
}
Student.prototype=new Person();
var s1=new Student();
var s2=new Student();
s1.myFriends.push("wangwu");
console.log(s1.myFriends);??
console.log(s2.myFriends);??
上面的代码我基于原型模式实现了一个简单的继承,那么上面的输出是什么呢?可以进行测试,发现输出全部是["zhangsna","lisi","wangwu"];为什么会出现这样的结果,我们不难分析出,我们为Student构造函数的原型重新指定了新的对象,那么此时这个对象(Person实例)就成了Student构造函数的原型对象,则其实例属性就变成了Student构造函数的原型属性,所以我们在通过Student实例访问其原型中引用类型属性 的时候会产生共享,从而出现上述输出。
这种输出结构和我们预期不符,如何处理这种问题呢,根据我们的查找规则,我们如果在本对象上查找到相对应的属性,不会去查找原型对象,基于这一点,我们只要覆盖原型上的对象即可,而最简单的方式就是借用构造函数,修改我们的代码如下:
function Person(){
this.name="zhangsan";
this.myFriends=["zhangsan","lisi"]
}
Person.prototype.getName=function(){
return this.name;
}
function Student(){
Person.call(this);//关键调用,注意这里是普通调用,非构造函数调用,本次调用,通过call改变this指向为Student的实例
}
Student.prototype=new Person();
var s1=new Student();
var s2=new Student();
s1.myFriends.push("wangwu");
console.log(s1.myFriends);
console.log(s2.myFriends);
使用一次普通的方法调用,传递this,然后动态的在this上创建name和myFriends属性,这样实例在输出时就不会查找原型上的属性了,解决了输出不符合预期的问题。
但是上面的代码仔细分析还有什么问题呢?
1)、Student.prototype.constructor 指向了Person
2)、调用了两次Person方法,一次是构造函数调用,一次是普通调用
3)、借用构造函数方式,是创建新的实例属性覆盖原型属性,这样会创建额外的属性。
再次修改我们的代码,处理上面的问题:
function inherit(child,parent){
function F(){}
F.prototype=parent.prototype;//只要原型上的内容,而不是实例,这样就避免了实例上的属性,只继承原型上的属性(方法)
F.prototype.constructor=child;//这里是强制修改constructor属性,让它指向子类型
child.prototype=F.prototype;
}
function Person(){
this.name="zhangsan";
this.myFriends=["zhangsan","lisi"]
}
Person.prototype.getName=function(){
return this.name;
}
function Student(){
Person.call(this);//关键调用,注意这里是普通调用,非构造函数调用,本次调用,通过call改变this指向为Student的实例
}
//Student.prototype=new Person();
inherit(Student,Person)
var s1=new Student();
var s2=new Student();
s1.myFriends.push("wangwu");
console.log(s1.myFriends);
console.log(s2.myFriends);
当然上述代码还是有些不完美,首先Student.prototype.constructor,应该是不可枚举的,我们这里是却可以。还有很多细节可能考虑不全,细节方面推荐大家啃一下《javascript高级程序设计》这本圣经,本篇就这么多吧。
- 事件流处理框架NEsper for .NET
- Quartz.net官方开发指南 第五课: SimpleTrigger
- SQL Server Performance Dashboard Reports
- 添加WordPress评论输入邮箱实时显示Gravatar头像功能
- Quartz.net官方开发指南 第六课 : CronTrigger
- WordPress 中禁止某个用户在线编辑主题
- Quartz.net官方开发指南 第七课 : TriggerListeners和JobListeners
- Quartz.net官方开发指南 第八课:SchedulerListeners
- 为WordPress 后台编辑器文本模式(HTML模式)添加按钮
- 360安全扫描之WordPress 页面异常导致本地路径泄漏 的漏洞修补
- Quartz.net官方开发指南 第九课: JobStore
- 数据中心运营中出现的错误
- Quartz.net官方开发指南 第十课: 配置、资源使用以及SchedulerFactory
- WCF服务上应用protobuf
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 最全Vue开发环境搭建
- apicloud api.openwin
- C#观察者模式实例
- xml序列化和反序列化(二)
- xml序列化和反序列化(一)
- js this问题和es6箭头函数this问题
- 元素淡入淡出效果实现
- Asp.Net Mvc表单提交(批量提交)
- Vue 基本指令和html常用标签结合使用综合案例(含代码)
- ef和mysql使用(二)--让mysql支持EntityFramework.Extended实现批量更新和删除
- EF 多种查询方式
- (mysql)找不到请求的 .Net Framework Data Provider。可能没有安装
- Vue 中的 v-cloak 解读
- ef和mysql使用(一)
- 浏览器工作原理