前端开发:面向对象与javascript中的面向对象实现(二)构造函数与原型
前言(题外话):
有人说拖延症是一个绝症,哎呀治不好了。先不说这是一个每个人都多多少少会有的,也不管它究竟对生活有多么大的影响,单单是自己的念想受到了一定得局限,想法不能够像平地而起的高楼大厦建成一样。可是那大楼也是有烂尾的呀,我觉得最重要的还是外在环境与个人观念的先决条件,决定了拖延症的症状的好坏,有那么一些人,它也有拖延症,但是它在拖的中间,想的更多,看的更远。事情在做的时候更加有条不紊,这拖延症这样看来,它也是好的吧。
1.原型:
javascript中没有提供传统的面向对象语言中的类(class)式继承,而是通过原型委托的方式来实现的对象与对象之间的继承,javascript也没有抽象类和接口,它在实现类或者设计模式的时候,原型模式提供了类似的方法。原型是用于创建对象的一种模式,可以理解为:“一个对象继承了另一个对象的属性和方法,那么就可以说被继承的对象是这个对象的原型”,更具体的说,就是,对象a通过某种方法引用对象b的属性,那么可以就可以说b是a的原型; 栗子:
/*克隆对象*/ function concat(obj){ if(obj instanceof Object){ if(Object.create!=undefined){ return Object.create(obj); //html5规范中新增Object.create()方法 }else{ var F = function(){ } F.prototype = obj; return new F(); } }else{ return -1; } } var A = { "name":"zhangtaifeng", "age":"21" } var B = concat(A); /*************对象B的原型是A*************/ B.job = "qianduankaifa"; alert(B.name);
上面的代码中,对象B通过concat方法克隆了一个对象A() 此时对象A是对象B的原型对象,而最后 B.job = "qianduankaifa"; 则是B对象的默认方法;我们分析控制台打印的结果:
html5规范中新增的Object.create()方法,它将A对象的引用指向了B对象的__proto__属性,__proto__即原型对象在javascript中的表现形式,在firefox控制台中打印可以看到,我们创建的每一个函数都有一个prototype原型属性,而这个对象的用途是包含可以由特定的类型的所有实例共享的属性和方法 上面这段要理解起来着实不容易,简单的说,javascript中所有的对象都有一个prototype的原型属性,它是用来保存多个对象共享的属性方法的。来个小例子:我们用来充电的插线板,你把手机通过充电器连接过去,开始充电,那么这个时候,我们如果把手机看成一个对象,这时候可以说插板是手机的充电原型。手机获得了插板的属性----电流;充电器就是他们中间的指针,连接的指针。这时候手机还是可以做其他操作的。实际上,每个对象默认的原型属性是从Obejct对象克隆继承来的,所以你可以说,对象B的原型是Object;
var A = { "name":"zhangtaifeng", "age":"21" } console.log(A.__proto__===Object.prototype); //true
这段代码最后返回了true; 说明,javascript所有的对象都默认继承了Object.prototype 所以对象可以调用Object提供的操作方法;
2.构造函数:
构造函数是一个创建对象的方法,通过new fn()的形式创建,它可以被看作类,但是它并不是类,而是函数构造器,javascript的函数既可以作为普通函数被调用,也可以作构造器被调用。当时用new操作符来调用函数时,这个函数就是一个构造函数,它会默认返回一个新的对象,用new运算符来创建对象的过程,实际上也是克隆Object.prototype对象,再进行一些其他操作的过程。
var Parse = function(){ this.name = "zhangtaifeng"; } var Parse01 = new Parse(); //到这一步 通过new操作符调用了Parse() 那么此时的Parse就是一个构造函数 alert(Parse01.name) //zhangtaifeng
构造函数命名首字母需大写,我们打印Parse01查看它的结果:
与上面的对象是同样的,它其实是创建了一个对象并赋值给了Parse01。
构造函数中可以直接设置他的原型对象属性:
var Parse = function(){ this.name = "zhangtaifeng"; } Parse.prototype = { "age":"21" } var Parse01 = new Parse(); alert(Parse01.age)
此时我们再次打印查看结果:
我们可以理解为,每个对象都有一个prototype原型属性,而构造函数的作用就是创建一个对象,只是它可以显示的通过Parse.prototype设置prototype的属性 此时Parse的原型是Parse.prototype这个对象。