JavaScript OOP(三):prototype原型对象(即构造函数的prototype属性)
function Obj(name,age){
this.name=name;
this.age=age;
this.func=function(){
return 'this is a test function';
};
}
var o1=new Obj('小明',10);
var o2=new Obj('小白',12);
console.log(o1.func===o2.func);
运行结果:

我们可以看出所有实例化对象(即o1,o2)中的func()都相同。但是每个对象都新建了func()方法,显得多余且浪费资源
function Obj1(name,age){
this.name=name;
this.age=age;
}
Obj1.prototype.address='beijing';//原型上的属性被所有实例化对象共享
Obj1.prototype.begin=function(){//原型上的方法被所有实例化对象共享
return 'this is a begin function';
};
var o3=new Obj1('晓明',100);
var o4=new Obj1('你猜',90);
console.log(o3,o4);
console.log(o3.address,o4.address);
console.log(o3.begin(),o4.begin());
运行结果:

o3.address='shanghai';
o3.begin=function(){
return 'this is a begin function for o3';
};
console.log(o3.address,o3.begin());
运行结果:

console.log(Object.getPrototypeOf(Object.prototype));
运行结果:

表明Object.prototype的原型对象是null
如果尝试获取null或undefined的原型对象:
console.log(Object.getPrototypeOf(null));//报错 Cannot convert undefined or null to object
注意:读取对象的某个属性时,js引擎会先在对象本身属性上寻找,如果找不到,那么去原型对象上找,一层一层往上"回溯",直至null.所以如果找寻某个不存在的属性,会将完整的原型链遍历一变,对性能影响较大。
constructor属性:prototype原型对象有一个constructor属性,默认指向prototype所在的构造函数
var O5=function(){};
O5.prototype=new Array();//O5的prototype属性等于new Array();那么O5的实例化对象共享着Array对象的所有属性和方法,并且O5.prototype的constructor也与Array.prototype的constructor一样
// O5.prototype.constructor=O5;
var o6=new O5();
o6.push(1,2,3);
console.log(o6);
console.log(O5.prototype.constructor===Array.prototype.constructor);//true
运行结果:

因为constructor属性定义在prototype上,所以所有实例对象都能访问
var a1=new O6();
var a2=new O6();
console.log(a1.constructor===a2.constructor && a1.constructor===O6.prototype.constructor);
运行结果:

var a3=new a2.constructor();
console.log(a3.constructor===a1.constructor);//true
注意原型对象被覆盖可能出现的问题:
function Abc(){};
Abc.prototype.constructor=Abc;
Abc.prototype.f=function(){
console.log('hello');
};
var a4=new Abc();
console.log(a4.constructor===Abc.prototype.constructor);
Abc.prototype={
f1:function(){
console.log('hi');
}
};//此时,定义Abc.prototype为一个新对象,原先定义在prototype上面的属性和方法均失效
//所以调用原先prototype的f方法,报错
var a5=new Abc();
//a5.f();//a5.f is not a function
//但是能调用新定义的属性或方法
a5.f1();//'hi'
运行结果:

function Obj2(){};
var o7=new Obj2();
console.log(o7.constructor.name);//Obj2
运行结果:

instanceof 关键字:判断对象是否为构造函数的实例
function Obj3(){};
var o8=new Obj3();
console.log(o8 instanceof Obj3);
console.log(o8 instanceof Object);
//instanceof对整个原型链上对象均有效
var n1=new Number(10);//var n1=new Object(10);
console.log(n1 instanceof Number,n1 instanceof Object);//true true
// o8 instanceof Obj3等同于Obj3.prototype.isPrototypeOf
console.log(Obj3.prototype.isPrototypeOf(o8));//true
//可以翻译成Obj3构造函数的prototype(原型)属性是o8的原型对象
//null,undefined instanceof Object均为false
运行结果:

//console.log(Object.getPrototypeOf(null),Object.getPrototypeOf(undefined));//Cannot convert undefined or null to object
不能获取null或者undefined的原型对象
function test1(){}
console.log(Object.getPrototypeOf(test1));
console.log(Object.getPrototypeOf(test1)===Function.prototype)
function Test(){}
var t1=new Test();
console.log(Object.getPrototypeOf(t1));
console.log(Object.getPrototypeOf(t1)===Test.prototype);
运行结果:

Object.setPrototypeOf():第一个参数是现有对象,第二个是原型对象;返回一个新对象
var a={
name:'apple',
f:function(){
console.log('this is a test function');
}
};
var o9=Object.setPrototypeOf({},a);
//o9本身是空对象,它的原型对象是a;o9能调用a的属性和方法
console.log(o9);
console.log(o9.name);//apple
o9.f();
运行结果:

function F(){};
var f1=new F();
console.log(f1);
/**
* 相当于
* var f1=Object.setPrototypeOf({},F.prototype);
* F.call(f1);//调用F函数,this指向f1
*/
var o10={
name:'chrome',
speed:'fast'
};
var o11=Object.create(o10);
console.log(o11);
console.log(o11.name,o11.speed);
console.log(Object.getPrototypeOf(o11)===o10);//true;说明o11的原型对象等于o10
运行结果:

console.log(o10.isPrototypeOf(o11));//true;o10是o11的原型对象
console.log(o11.isPrototypeOf(o10));
console.log(Object.prototype.isPrototypeOf(null));//除了null或者Object.create(null)生成对象,其它对象的原型对象往原型链回溯一定可以回溯到Object.prototype
console.log(Object.prototype.isPrototypeOf(Array));
运行结果:

var o12={
name:'apple',
age:100
};
var o13={
sex:'male',
email:'qq@com'
};
o12.__proto__=o13;
console.log(Object.getPrototypeOf(o12)===o13);
//即__pro__是指定对象的原型对象,也就是构造函数的prototype属性
运行结果:

总结:
- JavaScript的继承机制主要是基于prototype的。
- 构造函数生成实例化对象;构造函数的prototype属性就是实例化对象的原型对象;原型对象上的属性和方法被所有实例化对象所共享!
- 原型链一般往上回溯可以回溯到Object.prototype;Object.prototype的原型对象是null;而null的原型对象不存在!
- 原型对象上有construtor属性,等于构造函数名;因为是定义在原型对象上,所以被所有实例对象共享(由此我们也可以间接调用构造函数生成实例对象)!
- Object.getPrototypeOf():获取某个对象的原型;不能获取null或者undefined的原型
- Object.setPrototypeOf():第一个参数是现有对象,第二个是原型对象;返回一个新对象
- Object.create():以参数为原型对象生成新对象
- __proto__属性:设置对象的原型对象;尽量减少使用该属性
- instanceof:判断对象是否是某构造函数的实例对象
自己用Excel画的一张图:

JavaScript OOP(三):prototype原型对象(即构造函数的prototype属性)的更多相关文章
- JavaScript学习笔记之原型对象
本文是学习<JavaScript高级程序设计>第六章的笔记. JS中,便于批量创建对象的三种模式: 1.工厂模式:用一个函数封装创建对象的细节,传入必要的参数,在函数内部new一个对象并返 ...
- 理解js的prototype原型对象
我们创建的每一个函数都有一个prototype(原型)属性.这个属性是一个指针,指向一个对象,而这个对象的用途是包括能够由特定类型的全部实例共享的属性和方法.假设依照字面意思来理解,那么prototy ...
- 原型和原型对象(__proto__和prototype)转
看了之后我总算对原型继承有了更深刻的理解,做爱分享的姑娘,原文链接:理解Javascript 原型 我(个人)不喜欢的,就是讲原型时上来就拿类做比较的,所以我不会这样讲.不过我的确讲过构造器函数,在这 ...
- JavaScript 原型链学习(三)原型对象存在的问题 与 组合使用构造函数和原型
原型对象也不是没有缺点.首先,它省略了为构造函数传递初始化参数这一环节, 结果所有实例在默认情况下都将取得相同的属性值.虽然这会在某种程度上带来一些不方便, 但还不是原型对象的最大问题.原型对象的最大 ...
- 简单理解javascript中的原型对象,实现对之间共享属性和行为
javascript中提供了构造函数.可以方便的创建对象. 典型的构造函数例如以下: function Person(name, age) { this.name = name; this.age = ...
- JavaScript高级程序设计之原型对象
构造函数.原型对象.构造器是一体的关系,同时产生: 实例中的隐藏属性__proto__指向原型对象: 原型对象是这四种关系的纽带. 原型对象是动态的,不论在何处变化,实例中可以立即体现出来. var ...
- javascript(三):对象
对象(object)是javascript中很重要的数据类型.对象是“键值对”的集合,同时也是无序的.(注意:对象结尾处有分号) var ob1={ a1:'name',//a1可以加引号或者不加 a ...
- js之prototype 原型对象
原型对象prototype可以这么理解,是该类的实例对象的模板,每个实例对象都是先复制一份该类的prototype,通过这个可以让类的实例拥有相同的功能 String.prototype.say= ...
- js学习(四)- prototype原型对象
前言: 下面两行代码都是创建一个数组对象myArray:var myArray=[];//等价于var myArray=new Array();同样,下面的两段代码也都是创建一个函数myFunctio ...
随机推荐
- 2017年最受欢迎的UI框架
前端领域最近几年发展的特别迅速,可以说是百家争鸣.在底层的前端框架领域中,最早是jquery称霸互联网,近两年MVVM类型的框架慢慢成为主流,Vue.React和Angular三大框架并驾齐驱.可以说 ...
- url编码&&PHP大法
URL编码 Url编码通常也被称为百分号编码(Url Encoding,also known as percent-encoding),是因为它的编码方式非常简单,使用%百分号加上两位的字符--012 ...
- 数据库中WITH CHECK OPTION的用法
WITH CHECK OPTION:透过视图进行增删改操作时,不得破坏视图定义中的谓词条件(即子查询中的条件表达式) 例如: create view TestView as selec ...
- centos 源码安装python
一.准备环境 首先在官网下载想要的python对应版本http//www.python.org/downloads/source 下载tgz就可以了.文件有两种 1,Python-版本号.tgz(解压 ...
- Problem N
Problem Description The cows have purchased a yogurt factory that makes world-famous Yucky Yogurt. O ...
- c++学习笔记---05--- C++输出输入小结
C++输出输入小结 题目: 这个程序将向用户提出一个"Y/N"问题,然后把用户输入的值赋值给answer变量. 要求: 针对用户输入'Y'或'y'和'N'或'n'进行过滤: 发掘程 ...
- Android 开发笔记___DatePicker__日期选择器
虽然EditText提供了inputTtype="date",但用户往往不太喜欢自己输入时间. Android为这个提供了DatePicker,但有很多缺点,不是弹窗模式,而是直接 ...
- Problem F: 分数类的类型转换
Description 封装一个分数类Fract,用来处理分数功能和运算,支持以下操作: 1. 构造:传入两个参数n和m,表示n/m:分数在构造时立即转化成最简分数. 2. show()函数:分数 ...
- 详细图解window环境mongodb下载、安装、配置与使用
到官网下载最新版面mongodb安装包,(32位版本的已经取消了,只有64位的) 官网地址: https://www.mongodb.com/download-center#community 下载完 ...
- 基于python3.x,使用Tornado中的torndb模块操作数据库
目前Tornado中的torndb模块是不支持python3.x,所以需要修改部分torndb源码即可正常使用 1.开发环境介绍 操作系统:win8(64位),python版本:python3.6(3 ...