JavaScript原型与原型链分析

2017-03-13 12:07:19

JavaScript没有类的概念,但几乎所有的东西又是基于对象的,同时也能实现继承,这就是js跟其他OOP语言最大的不同之处,这也是js最难理解的一块。下面我来说说我个人的理解。

首先从创建对象说起,一般会有下面几种方法:

1.创建一个Object实例,然后给它添加属性和方法。

var person() = new Object();
person.name = 'mikej';
person.sayName = function(){
  alert(this.name);
}

2.也可以这样写:

var parson = {
  name : 'mikej',
  sayName : function(){
    alert(this.name);
  }
}

3.这两种创建对象的方法很简单,但都有缺陷,使用同一个模式创建对象的时候,会产生大量重复代码。于是就有了工厂模式:

function createPerson(name){  var p = new Object();
  p.name=name;
  p.sayName = function(){
    alert(this.name);
  };  return p;
}var p1 = createPerson('mikej');var p2 = createPerson('tom');

这样就可以无限创建对象了。

4.还有一种方法,跟工厂模式异曲同工,叫做构造函数模式:

function Person(name){  this.name=name  this.sayName = function(){
   alert(this.name);
  }  this.say = function(){
    alert('hi');
  }
}var p1 = new Person('mikej');var p2 = new Person('tom');

这里有几个值得关注的地方:没有显示的创建对象、函数名Person使用的是大写字母P(这是必须的)、p1和p2中都有一个constructor(构造函数)属性,指向Person。同时p1和p2既是Object的实例,也是Person的实例。

alert(p1.constructor == Person); //truealert(p1 instanceof Object); //truealert(p1 instanceof Person); //true

//5.11更新:以一个phper的角度看的话,之前很容易将创建对象的流程想成这样,Person就是一个“类”,然后用new Person('mikej')实例化了这个类,并且传入参数。但实际上并不是这样的,创建的流程应该是这样:首先,创建一个空对象,然后用apply方法,第一个参数是这个空对象,第二个参数是上下文的参数,这样Person中的this就会指向这个对象,也就是p1。

var p1 = new Person('mikej');//上面代码就相当于var p1 = {};
Person.apply(p1, ['mikej']);

构造函数模式看上去很好,但是它有一个弊端就是浪费内存,接上例

alert(p1.say == p2.say) //false

.为了避免这个缺陷,可是使用原型模式来创建对象,js中的每个对象都有一个prototype属性用来指向另外一个对象,这个对象的所有属性和方法都会被构造函数的实例继承,是共享的,这就意味着,我们可以把那些不变的属性和方法,定义到prototype对象上。

function Person(name){  this.name = name;
}//Person的原型对象Person.prototype = {
  say: function(){
    alert('hi');
  },
  sayName: function(){
    alert(this.name);
  }
};var p1 = new Person("mikej");var p2 = new Person("tom");
p1.sayName();
p2.sayName();//下面就可以看出方法实现了共享alert(P1.say == P2.say) //truealert(P1.sayName == P2.sayName) //true

再来扩展一下上面的例子,使用原型来实现继承。

function Person(name){  this.name = name;
}

Person.prototype = {
  say: function(){
    alert('hi');
  },
  sayName: function(){
    alert(this.name);
  }
};function Programmer(){  this.say = function(){
    alert('im Programmer, my name is ' + this.name);
  }
}

Programmer.prototype = new Person('mikej');//手动修正构造函数Programmer.prototype.constructor = Programmer;var p1 = new Programmer();console.dir(Programmer.prototype.constructor);//Programmerconsole.dir(p1.constructor);//Programmerconsole.dir(p1);

Programmer的原型指向了Person的一个实例,那么所有的Programmer的实例都能继承Person和Person的原型了。

这里会有一个问题。

默认原型对象里有一个constructor属性,指向它的构造函数。而每一个实例也有一个constructor属性,会默认调用prototype对象的constructor属性。

假设没有Programmer.prototype = new Person('mikej');

Programmer.prototype.constructor是指向Programmer的。p1的构造也指向Programmer

alert(Programmer.prototype.constructor == Programmer) //truealert(p1.constructor == Programmer) //true

但有了这句Programmer.prototype = new Person('mikej');之后,

Programmer.prototype.constructor就指向了Object,也就是Person.prototype指向的对象的构造。p1.constructor也指向了Object。但p1明明是构造函数Programmer生成的,这就造成了继承的混乱,所以我们必须手动修正构造函数,也就是下面这代码。

Programmer.prototype.constructor = Programmer;

好了,现在我们再来看看原型链:

console.dir(p1);

这句代码的结果是

prototype

可以看出,p1是Programmer的实例,Programmer的原型是Person,Person的原型是Object,再网上就是JS的对象了。这就是原型链,也就是说,JavaScript的继承是基于原型链来实现的。


  • 2020-04-17 09:27:38

    推荐一个老前端开发者的博客

    前端修炼场,首页标签大全greenSock前端研究VUE研究我们的作品flash技术探讨开发心得个人档案培训与招聘服务报价

  • 2020-04-17 09:41:47

    前端css博客推荐

    这个博客有大量的css内容,有svg,TweenMax等教程,抽空通读一下

  • 2020-04-17 10:20:47

    GreenSocks Animation Platform详细工作机制以及TweenMax用法

    GSAP(GreenSocks Animation Platform)是一个性能较好的前端动画库。最近在写一个前端SVG动画编辑器时选择了它作为底层的动画库。为了减少踩坑,我大致浏览了它的源代码,这篇文章主要是对我的理解进行记录。 我会先简单介绍一下这个动画库的API,再介绍它的插件机制,最后会从一个用例出发跟踪其运行机制。

  • 2020-04-17 10:39:02

    CSS 滤镜技巧与细节,实现火焰,融合等特效

    简单来说,CSS 滤镜就是提供类似 PS 的图形特效,像模糊,锐化或元素变色等功能。通常被用于调整图片,背景和边界的渲染。本文就会围绕这些滤镜展开,看看具体能怎么使用或者玩出什么花活。

  • 2020-04-17 10:42:29

    (三)TweenMax运动效果

    运动效果 实例化对象.set() 立刻运动到指定地点,不用加时间

  • 2020-04-17 11:19:55

    Vue中的is和操作DOM

    vue中is的属性引入是为了解决dom结构中对放入html的元素有限制的问题,譬如ul里面要接上li的标签,引入is的属性后,你完全可以写成这样

  • 2020-04-17 11:27:48

    TweenMax中文初级教程一

    TweenMax.js集成了GreenSock动画平台的大部分核心功能,且具有极高的兼容性。

  • 2020-04-17 11:28:35

    TweenMax中文初级教程二

    TimelineMax是GreenSock 动画平台中的动画组织、排序、管理工具,可创建时间轴(timeline)作为动画或其他时间轴的容器,这使得整个动画控制和精确管理时间变得简单,避免了通过反复delay和回调进行动画。 作者:李霖弢 链接:https://www.jianshu.com/p/8c0361e43bf5 来源:简书 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  • 2020-04-17 11:28:57

    TweenMax中文初级教程三

    动画关键词:CSS(一般可以省略) CSSPlugin用于在TweenMax中对DOM元素的CSS相关属性进行动画 在CSSPlugin中CSS属性需要写成驼峰式,例如font-size应当写作fontSize。有时候你可以在一些默认px为单位的属性中省略单位,CSSPlugin还可以在不同的单位间做动画:

  • 2020-04-17 11:29:23

    TweenMax中文初级教程四

    用于滚动窗口(类似于window.scrollTo(x, y))或DOM元素(如myDiv.scrollTop = y; myDiv.scrollLeft = x;)。滚动窗口时使用window作为动画目标。