Js中Prototype、__proto__、Constructor、Object、Function关系介绍 ,JS原型

此文来自:http://www.blogjava.net/heavensay/archive/2013/10/20/405440.html

Js中Prototype、__proto__、Constructor、Object、Function关系介绍

一    Prototype、__proto__与Object、Function关系介绍

        Function、Object:Js自带的函数对象。

prototype,每一个函数对象都有一个显示的prototype属性,它代表了对象的原型(Function.prototype函数对象是个例外,没有prototype属性)。

__proto__:每个对象都有一个名为__proto__的内部隐藏属性,指向于它所对应的原型对象(chrome、firefox中名称为__proto__,并且可以被访问到)。原型链正是基于__proto__才得以形成(note:不是基于函数对象的属性prototype)。

关于上面提到的函数对象,我们来看以下例子,来说明:

var o1 = {};
        var o2 =new Object();
        
        function f1(){}
        var f2 = function(){}
        var f3 = new Function(‘str‘,‘console.log(str)‘);
    
        f3(‘aabb‘);   // aabb
        console.log(‘typeof Object:‘+typeof Object);            //function
        console.log(‘typeof Function:‘+typeof Function);        //function
        console.log(‘typeof o1:‘+typeof o1);   //object
        console.log(‘typeof o2:‘+typeof o2);   //object
        console.log(‘typeof f1:‘+typeof f1);   //function
        console.log(‘typeof f2:‘+typeof f2);   //function
        console.log(‘typeof f3:‘+typeof f3);   //function

  • 通常我们认为o1、o2是对象,即普通对象;f1、f2、f3为函数。
  • 但是其实函数也是对象,是由Function构造的,
  • f3这种写法就跟对象的创建的写法一样。f1、f2最终也都像f3一样是有Function这个函数构造出来的
  • f1、f2、f3为函数对象,Function跟Object本身也是函数对象。

Js中每个对象(null除外)都和另一个对象相关联,通过以下例子跟内存效果图来分析Function、Object、Prototype、__proto__对象间的关系。

function Animal(){
        
    }
    var  anim = new Animal();
    
    console.log(‘***********Animal anim proto*****************‘);
    console.log(‘typeof Animal.prototype:‘ +typeof Animal.prototype);  //object 
    console.log(‘anim.__proto__===Animal.prototype:‘+(anim.__proto__===Animal.prototype));  //true
    console.log(‘Animal.__proto__===Function.prototype:‘+(Animal.__proto__===Function.prototype));  //true
    console.log(‘Animal.prototype.__proto__===Object.prototype:‘+(Animal.prototype.__proto__===Object.prototype));  //true
    
    console.log(‘***********Function proto*****************‘);
    console.log(‘typeof Function.prototype:‘+typeof Function.prototype);  //function
    console.log(‘typeof Function.__proto__:‘+typeof Function.__proto__);  //function
    console.log(‘typeof Function.prototype.prototype:‘+typeof Function.prototype.prototype); //undefined
    console.log(‘typeof Function.prototype.__proto__:‘+typeof Function.prototype.__proto__);   //object
    console.log(‘Function.prototype===Function.__proto__:‘+(Function.prototype===Function.__proto__)); //true

console.log(‘***********Object proto*****************‘);
    console.log(‘typeof Object.prototype:‘+typeof Object.prototype);  //object
    console.log(‘typeof Object.__proto__:‘+typeof Object.__proto__);  //function
    console.log(‘Object.prototype.prototype:‘+Object.prototype.prototype);  //undefied
    console.log(‘Object.prototype.__proto__===null:‘+(Object.prototype.__proto__===null));  //null

console.log(‘***********Function Object  proto关系*****************‘);
    console.log(‘Function.prototype===Object.__proto__:‘+(Function.prototype===Object.__proto__));   //true
    console.log(‘Function.__proto__===Object.__proto__:‘+(Function.__proto__===Object.__proto__));   //true
    console.log(‘Function.prototype.__proto__===Object.prototype:‘+(Function.prototype.__proto__===Object.prototype));   //true

/********************* 系统定义的对象Array、Date ****************************/
    console.log(‘**************test Array、Date****************‘);      
    var array = new Array();
    var date = new Date();
    console.log(‘array.__proto__===Array.prototype:‘+(array.__proto__===Array.prototype));   //true
    console.log(‘Array.__proto__===Function.prototype:‘+(Array.__proto__===Function.prototype));  //true
    console.log(‘date.__proto__===Date.prototype:‘+(date.__proto__===Date.prototype));    //true
    console.log(‘Date.__proto__===Function.prototype:‘+(Date.__proto__===Function.prototype));     //true

Function、Object、Prototype、__proto__内存关系图

上面的内存图跟堆栈结构可以参照文章Javascript_01_理解内存分配
        堆区图说明:
 
        Function.prototype函数对象图内部表示prototype属性的红色虚框,只是为了说明这个属性不存在。

通过上图Function、Object、Prototype关系图中,可以得出一下几点:

  1. 所有对象所有对象,包括函数对象的原型链最终都指向了Object.prototype,而Object.prototype.__proto__===null,原型链至此结束。
  2. Animal.prototype是一个普通对象。
  3. Object是一个函数对象,也是Function构造的,Object.prototype是一个普通对象。
  4. Object.prototype.__type__指向null。
  5. Function.prototype是一个函数对象,前面说函数对象都有一个显示的prototype属性,但是Function.prototype却没有prototype属性,即Function.prototype.prototype===undefined,所有Function.prototype函数对象是一个特例,没有prototype属性。
  6. Object虽是Function构造的一个函数对象,但是Object.prototype没有指向Function.prototype,即Object.prototype!==Function.prototype。


二    Prototype跟Constructor关系
介绍

在 JavaScript 中,每个函数对象都有名为“prototype”的属性(上面提到过Function.prototype函数对象是个例外,没有prototype属性),用于引用原型对象。此原型对象又有名为“constructor”的属性,它反过来引用函数本身。这是一种循环引用(i.e. Animal.prototype.constructor===Animal)。

通过以下例子跟内存效果图来分析Prototype、constructor间的关系。

console.log(‘**************constructor****************‘);

console.log(‘anim.constructor===Animal:‘+(anim.constructor===Animal))    ;    //true
    console.log(‘Animal===Animal.prototype.constructor:‘+(Animal===Animal.prototype.constructor))    ;    //true
    console.log(‘Animal.constructor===Function.prototype.constructor:‘+(Animal.constructor===Function.prototype.constructor));   //true
    console.log(‘Function.prototype.constructor===Function:‘+(Function.prototype.constructor===Function));    //true
    console.log(‘Function.constructor===Function.prototype.constructor:‘+(Function.constructor===Function.prototype.constructor));    //true

console.log(‘Object.prototype.constructor===Object:‘+(Object.prototype.constructor===Object));    //true
    console.log(‘Object.constructor====Function:‘+(Object.constructor===Function));    //true

 prototype、constructor内存关系图(在Function、Object、Prototype关系图上加入constructor元素):

上图中,红色箭头表示函数对象的原型的constructor所指向的对象。

  1. 注意Object.constructor===Function;本身Object就是Function函数构造出来的
  2. 如何查找一个对象的constructor,就是在该对象的原型链上寻找碰到的第一个constructor属性所指向的对象。

参考:

http://www.cnblogs.com/fool/category/264215.html (javascript原理介绍)

http://www.libuchao.com/2012/05/14/prototypes-in-javascript/ (JavaScript 的原型对象 Prototype)

http://rockyuse.iteye.com/blog/1426510 (理解js中的原型链,prototype与__proto__的关系)

时间: 2024-10-14 15:37:28

Js中Prototype、__proto__、Constructor、Object、Function关系介绍 ,JS原型的相关文章

Js中Prototype、__proto__、Constructor、Object、Function关系介绍

Js中Prototype.__proto__.Constructor.Object.Function关系介绍 一    Prototype.__proto__与Object.Function关系介绍        Function.Object:Js自带的函数对象. prototype,每一个函数对象都有一个显示的prototype属性,它代表了对象的原型(Function.prototype函数对象是个例外,没有prototype属性). __proto__:每个对象都有一个名为__proto

Object.prototype.__proto__ 、Object.prototype和 Object.prototype.constructor

Object.prototype.__proto__: 实体对象指向造它的构造函数的 prototype属性所指 的对象 ,实例的__proto__是引用构造函数的prototype属性所指对象, Object.prototype : js规定,构造函数有prototype 属性,指向一个对象,这个对象一般就是构造函数的公有成员. 所以 __proto__  和 prototype 可以指向同一个对象  {key1:value1,key2:value2,......},构造函数的私有和特权函数会

js中prototype用法(转)

JavaScript能够实现的面向对象的特征有:·公有属性(public field)·公有方法(public Method)·私有属性(private field)·私有方法(private field)·方法重载(method overload)·构造函数(constructor)·事件(event)·单一继承(single inherit)·子类重写父类的属性或方法(override)·静态属性或方法(static member) 例子一(JavaScript中允许添加行为的类型):可以在

js 中使用el表达式 关键总结:在js中使用el表达式一定要使用双引号

js 中使用el表达式 关键总结:在js中使用el表达式一定要加双引号 js控制中用到了el表达式,最开始源码如下: Java代码   var selected = ${requestScope.xxxxForm.recordNumPerPage} 这样始终js错误,因为在第一次的时候requestScope.xxxxForm.recordNumPerPage为null.于是在boss的指导下,简单的加了两个引号,代码变成 Java代码   var selected = "${requestSc

js中prototype,__proto__,constructor之间的关系

首先,我们需要了解三点: 1. 只要创建一个任意新函数,就会根据一个prototype属性,该属性指向函数的原型对象: 2. 每一个原型对象都会自动获得一个constructor属性,该属性只想prototype所在函数的指针: 3. 当调用构造函数创建实例时,该实例内部将包含一个指向构造函数原型对象的指针,在大部分浏览器中用__proto__标识: 从上面这三点我们可以了解到: 正常情况下:prototype属性 -> 原型对象 原型对象的constructor属性 -> 构造函数(即拥有p

js中prototype与__proto__的关系详解

一.构造函数: 构造函数:通过new关键字可以用来创建特定类型的对象的函数.比如像Object和Array,两者属于内置的原生的构造函数,在运行时会自动的出现在执行环境中,可以直接使用.如下: var arr = new Array();//使用Array构造函数创建了一个array实例arr arr[0]="a"; arr[1]="b"; alert(arr);//a,b var obj=new Object();//使用Object构造函数创建了一个Object

实践一些js中的prototype, __proto__, constructor

<!DOCTYPE html> <html> <head> <title>ExtJs</title> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> <link rel="stylesheet" type="text/css" href="ExtJs/

Bom和Dom编程以及js中prototype的详解

一.Bom编程: 1.事件练习: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <meta http-equ

原型链之prototype/__proto__/constructor

(一)prototype 每个函数都有一个prototype属性,这个属性是指向一个对象的引用,这个对象称为原型对象,原型对象包含函数实例共享的方法和属性,也就是说将函数用作构造函数调用(使用new操作符调用)的时候,新创建的对象会从原型对象上继承属性和方法. 通过Function.prototype.bind方法构造出来的函数是个例外,它没有prototype属性 (二)__proto__ 根据ECMA定义 'to the value of its constructor’s "prototy