首页 web前端 js教程 学习javascript面向对象 理解javascript原型和原型链_javascript技巧

学习javascript面向对象 理解javascript原型和原型链_javascript技巧

May 16, 2016 pm 03:22 PM
javascript 原型 原型链

先看一张图,梳理梳理。

一、基本概念  
【原型链】每个构造函数都有一个对象,原型对象都包含一个指向构造函数的指针,而实例都包含一个指向原型对象的内部指针。那么,如果原型对象等于另一个原型的实例,此时的原型对象将包含一个指向另一个原型的指针,相应地,另一个原型中也包含着一个指向另一个构造函数的指针。如果另一个原型又是另一个原型的实例,那么上述关系依然成立。如此层层递进,就构成了实例与原型的链条。
【原型对象】这个对象包含可以由特定类型的所有实例共享的属性和方法。所有引用类型默认都继承了Object,而这个继承也是通过原型链实现的。所有函数的默认原型都是Object的实例,因此默认原型都会包含一个内部指针,指向Object.prototype,这也正是所有自定义类型都会继承toString()、valueOf()方法的原因
【构造函数】构造函数与其他函数的区别在于调用它们的方式不同。一般来说,函数只要通过new操作符来调用,那它就可以作为构造函数;如果不通过new操作符来调用,那它跟普通函数也不会有什么两样。
[注意]用户自定义的函数和javascript中内置的构造函数可以当成构造函数使用
【构造函数的写法】构造函数始终应该以一个大写字母开头,而非构造函数以一个小写字母开头。这个做法借鉴自其他OO语言,主要是为了区别于ECMAScript中的其他函数;因为构造函数本身也是函数,只不过可以用来创建对象而已
【构造函数的三种使用场景】
[a]当作构造函数使用

1

2

var person = new Person("Nicholas",29,"software Engineer");

person.sayName();

登录后复制

[b]当作普通函数调用

1

2

Person("greg",27,"doctor");//添加到window

window.sayName();//"Greg"

登录后复制

[c]在另一个对象的作用域中调用

1

2

3

var o = new Object();

Person.call(o,"Kristen",25,"Nurse");

o.sayName();//"Kristen"

登录后复制

【prototype属性】只要创建了一个新函数,就会根据一组特定的规则为该函数创建一个prototype属性,这个属性指向函数的原型对象。
[注意]只有函数才有prototype属性,object没有prototype属性
【constructor属性】在默认情况下,所有原型对象都会自动获得一个constructor(构造函数)属性,这个属性包含一个指向prototype属性所在函数的指针
[注意]创建了自定义的构造函数之后,其原型对象默认只会取得constructor属性,至于其他方法则都是从Object继承而来的
【_proto_和[[prototype]]】当调用构造函数创建一个新实例后,该实例的内部将包含一个指针(内部属性),指向构造函数的原型对象。ECMA-262第5版管这个指针叫[[prototype]]。虽然在脚本中标准的方式访问[[prototype]],但firefox\safari\chrome在每个对象上都支持一个属性_proto_;而在其他实现中,这个属性对脚本则是完全不可见的。这个连接存在于实例与构造函数的原型对象之间,而不是存在于实例与构造函数之间
二、基本操作  
【原型链查询】每当代码读取某个对象的某个属性时,都会执行一次搜索,目标是具有给定名字的属性。搜索首先从对象实例本身开始,如果在实例中找到了具有给定名字的属性,则返回该属性的值;如果没有找到,则继续搜索指针指向的原型对象,在原型对象中查找具有给定名字的属性,如果找到了这个属性,则返回该属性的值。
【添加实例属性】当为对象实例添加一个属性时,这个属性就会屏蔽原型对象中保存的同名属性;换句话说,添加这个属性只会阻止我们访问原型中的那个属性,但不会修改那个属性,即使将这个属性设置为null,也只会在实例中设置这个属性,而不会恢复其指向原型的连接。不过,使用delete操作符则可以完全删除实例属性,从而让我们能够重新访问原型中的属性。
【原型的动态性】由于在原型中查找值的过程是一次搜索,因此我们对原型对象所做的任何修改都能立即从实例上反映出来,即使是先创建了实例后修改原型也照样如此。
[注意]不推荐在产品化的程序中修改原生对象的原型

1

2

3

4

5

6

function Person(){};

var friend = new Person();

Person.prototype.sayHi = function(){

  alert('hi');

}

friend.sayHi();//"hi" 

登录后复制

【重写原型】调用构造函数时会为实例添加一个指向最初原型的[[prototype]]指针,而把原型修改为另外一个对象就等于切断了构造函数与最初原型之间的联系。实例中的指针仅指向原型,而不指向构造函数。
三、基本方法  
[1]isPrototypeOf():判断实例对象和原型对象是否存在于同一原型链中,只要是原型链中出现过的原型,都可以说是该原型链所派生的实例的原型

1

2

3

4

5

6

7

function Person(){};

var person1 = new Person();

var person2 = new Object();

console.log(Person.prototype.isPrototypeOf(person1));//true

console.log(Object.prototype.isPrototypeOf(person1));//true

console.log(Person.prototype.isPrototypeOf(person2));//false

console.log(Object.prototype.isPrototypeOf(person2));//true

登录后复制

[2]ECMAScript5新增方法Object.getPrototypeOf():这个方法返回[[Prototype]]的值

1

2

3

4

5

6

7

function Person(){};

var person1 = new Person();

var person2 = new Object();

console.log(Object.getPrototypeOf(person1)); //Person{}

console.log(Object.getPrototypeOf(person1) === Person.prototype); //true

console.log(Object.getPrototypeOf(person1) === Object.prototype); //false

console.log(Object.getPrototypeOf(person2)); //Object{} 

登录后复制

[3]hasOwnProperty():检测一个属性是否存在于实例中

1

2

3

4

5

6

7

8

9

10

11

12

13

14

function Person(){

  Person.prototype.name = 'Nicholas';

}

var person1 = new Person();

//不存在实例中,但存在原型中

console.log(person1.hasOwnProperty("name"));//false

//不存在实例中,也不存在原型中

console.log(person1.hasOwnProperty("no"));//false

person1.name = 'Greg';

console.log(person1.name);//'Greg'

console.log(person1.hasOwnProperty('name'));//true

delete person1.name;

console.log(person1.name);//"Nicholas"

console.log(person1.hasOwnProperty('name'));//false 

登录后复制

[4]ECMAScript5的Object.getOwnPropertyDescriptor():只能用于取得实例属性的描述符,要取得原型属性的描述符,必须直接在原型对象上调用Object.getOwnPropertyDescription()方法

1

2

3

4

5

6

7

function Person(){

  Person.prototype.name = 'Nicholas';

}

var person1 = new Person();

person1.name = 'cook';

console.log(Object.getOwnPropertyDescriptor(person1,"name"));//Object {value: "cook", writable: true, enumerable: true, configurable: true}

console.log(Object.getOwnPropertyDescriptor(Person.prototype,"name"));//Object {value: "Nicholas", writable: true, enumerable: true, configurable: true}

登录后复制

[5]in操作符:在通过对象能够访问给定属性时返回true,无论该属性存在于实例还是原型中

1

2

3

4

5

6

7

8

9

function Person(){}

var person1 = new Person();

person1.name = 'cook';

console.log("name" in person1);//true

console.log("name" in Person.prototype);//false

var person2 = new Person();

Person.prototype.name = 'cook';

console.log("name" in person2);//true

console.log("name" in Person.prototype);//true

登录后复制

[6]同时使用hasOwnProperty()方法和in操作符,来确定属性是否存在于实例中

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

//hasOwnProperty()返回false,且in操作符返回true,则函数返回true,判定是原型中的属性

function hasPrototypeProperty(object,name){

  return !object.hasOwnProperty(name) && (name in object);

}

function Person(){

  Person.prototype.name = 'Nicholas';

}

var person1 = new Person();

console.log(hasPrototypeProperty(person1,'name'));//true

person1.name = 'cook';

console.log(hasPrototypeProperty(person1,'name'));//false

delete person1.name;

console.log(hasPrototypeProperty(person1,'name'));//true

delete Person.prototype.name;

console.log(hasPrototypeProperty(person1,'name'));//false

登录后复制

[7]ECMAScript5的Object.keys()方法:接收一个对象作为参数,返回一个包含所有可枚举属性的字符串数组
[注意]一定要先new出实例对象再使用该方法,否则为空

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

function Person(){

  Person.prototype.name = 'Nicholas';

  Person.prototype.age = 29;

  Person.prototype.job = 'Software Engineer';

  Person.prototype.sayName = function(){

    alert(this.name);

  

};

var keys = Object.keys(Person.prototype);

console.log(keys);//[]

var p1 = new Person();

p1.name = "Rob";

p1.age = 31;

var keys = Object.keys(Person.prototype);

console.log(keys);//["name","age","job","sayName"]

var p1Keys = Object.keys(p1);

console.log(p1Keys);//["name","age"]

登录后复制

[8]ECMAScript5的Object.getOwnPropertyNames()方法:接收一个对象作为参数,返回一个包含所有属性的字符串数组
[注意]一定要先new出实例对象再使用该方法,否则只有constructor

1

2

3

4

5

6

7

8

9

10

11

12

13

function Person(){

  Person.prototype.name = 'Nicholas';

  Person.prototype.age = 29;

  Person.prototype.job = 'Software Engineer';

  Person.prototype.sayName = function(){

    alert(this.name);

  

};

var keys = Object.getOwnPropertyNames(Person.prototype);

console.log(keys);//["constructor"]

var p1 = new Person();

var keys = Object.getOwnPropertyNames(Person.prototype);

console.log(keys);//["constructor", "name", "age", "job", "sayName"]

登录后复制

希望本文所述对大家学习javascript程序设计有所帮助。

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

如何使用WebSocket和JavaScript实现在线语音识别系统 如何使用WebSocket和JavaScript实现在线语音识别系统 Dec 17, 2023 pm 02:54 PM

如何使用WebSocket和JavaScript实现在线语音识别系统引言:随着科技的不断发展,语音识别技术已经成为了人工智能领域的重要组成部分。而基于WebSocket和JavaScript实现的在线语音识别系统,具备了低延迟、实时性和跨平台的特点,成为了一种被广泛应用的解决方案。本文将介绍如何使用WebSocket和JavaScript来实现在线语音识别系

WebSocket与JavaScript:实现实时监控系统的关键技术 WebSocket与JavaScript:实现实时监控系统的关键技术 Dec 17, 2023 pm 05:30 PM

WebSocket与JavaScript:实现实时监控系统的关键技术引言:随着互联网技术的快速发展,实时监控系统在各个领域中得到了广泛的应用。而实现实时监控的关键技术之一就是WebSocket与JavaScript的结合使用。本文将介绍WebSocket与JavaScript在实时监控系统中的应用,并给出代码示例,详细解释其实现原理。一、WebSocket技

如何利用JavaScript和WebSocket实现实时在线点餐系统 如何利用JavaScript和WebSocket实现实时在线点餐系统 Dec 17, 2023 pm 12:09 PM

如何利用JavaScript和WebSocket实现实时在线点餐系统介绍:随着互联网的普及和技术的进步,越来越多的餐厅开始提供在线点餐服务。为了实现实时在线点餐系统,我们可以利用JavaScript和WebSocket技术。WebSocket是一种基于TCP协议的全双工通信协议,可以实现客户端与服务器的实时双向通信。在实时在线点餐系统中,当用户选择菜品并下单

如何使用WebSocket和JavaScript实现在线预约系统 如何使用WebSocket和JavaScript实现在线预约系统 Dec 17, 2023 am 09:39 AM

如何使用WebSocket和JavaScript实现在线预约系统在当今数字化的时代,越来越多的业务和服务都需要提供在线预约功能。而实现一个高效、实时的在线预约系统是至关重要的。本文将介绍如何使用WebSocket和JavaScript来实现一个在线预约系统,并提供具体的代码示例。一、什么是WebSocketWebSocket是一种在单个TCP连接上进行全双工

JavaScript和WebSocket:打造高效的实时天气预报系统 JavaScript和WebSocket:打造高效的实时天气预报系统 Dec 17, 2023 pm 05:13 PM

JavaScript和WebSocket:打造高效的实时天气预报系统引言:如今,天气预报的准确性对于日常生活以及决策制定具有重要意义。随着技术的发展,我们可以通过实时获取天气数据来提供更准确可靠的天气预报。在本文中,我们将学习如何使用JavaScript和WebSocket技术,来构建一个高效的实时天气预报系统。本文将通过具体的代码示例来展示实现的过程。We

简易JavaScript教程:获取HTTP状态码的方法 简易JavaScript教程:获取HTTP状态码的方法 Jan 05, 2024 pm 06:08 PM

JavaScript教程:如何获取HTTP状态码,需要具体代码示例前言:在Web开发中,经常会涉及到与服务器进行数据交互的场景。在与服务器进行通信时,我们经常需要获取返回的HTTP状态码来判断操作是否成功,根据不同的状态码来进行相应的处理。本篇文章将教你如何使用JavaScript获取HTTP状态码,并提供一些实用的代码示例。使用XMLHttpRequest

JavaScript和WebSocket:打造高效的实时图像处理系统 JavaScript和WebSocket:打造高效的实时图像处理系统 Dec 17, 2023 am 08:41 AM

JavaScript是一种广泛应用于Web开发的编程语言,而WebSocket则是一种用于实时通信的网络协议。结合二者的强大功能,我们可以打造一个高效的实时图像处理系统。本文将介绍如何利用JavaScript和WebSocket来实现这个系统,并提供具体的代码示例。首先,我们需要明确实时图像处理系统的需求和目标。假设我们有一个摄像头设备,可以采集实时的图像数

如何在JavaScript中获取HTTP状态码的简单方法 如何在JavaScript中获取HTTP状态码的简单方法 Jan 05, 2024 pm 01:37 PM

JavaScript中的HTTP状态码获取方法简介:在进行前端开发中,我们常常需要处理与后端接口的交互,而HTTP状态码就是其中非常重要的一部分。了解和获取HTTP状态码有助于我们更好地处理接口返回的数据。本文将介绍使用JavaScript获取HTTP状态码的方法,并提供具体代码示例。一、什么是HTTP状态码HTTP状态码是指当浏览器向服务器发起请求时,服务

See all articles