“javascript中的prototype真是太好用了,在我理解prototype的时候感觉自己对js有了顿悟!”小白欢喜的说道。
老朱:“恩,其实当你能把javascript中所有的数据类型都能理解为对象(包括函数也可以看作对象)之后,你再重新审视js就会发现它原来是这么灵活,以前很多公共js控件看不懂的代码突然能够理解了。”
小白:“是啊!我在没系统学js之前一直以为js是一种简单的面向过程的脚本语言,自从接触对象之后才发现页面中每一个标签都是对象,按照面向对象的方式会让我从更高的角度对标签进行思考。”
老朱:“不过小白你得做好心理准备,js不像java、c#的面向对象严禁,c#中我们可以创建不同的类放到对应的包中,需要哪个类通过包路径引入就可以,js中要把类归类还需要一些特殊的方法才能实现。另外c#中继承一个类很简单,js里面却需要通过一些特殊方法才能实现。所以你还得做好继续深入学习的准备。”
小白斩钉截铁的说道:“没问题,我一定继续努力。朱哥,你刚说js也能做继承,这是怎么实现的呢?”
老朱:“把你昨天做的那个男人类(Man)代码给我,我跟你大概聊聊怎么继承。”
老朱拿到代码后去了几行没用的代码,跟小白说道:“你看现在这个男人类,公共变量是姓名(name)、电话(tel),私有变量是性别,公共方法是获取性别信息(sexInfo)和去哪儿(towhere)。如果我们要实例化它只需要使用new关键字就可以了。”
小白:“恩,对!那应该怎么继承呢?”
老朱:“我们创建一个老男人类(OldMan),让OldMan的原型prototype指向Man的一个实例就可以实现。”
小白:“为什么不是指向Man这个类而是指向它的实例呢?”
老朱:“还记得昨天我们定义一个类以后,直接通过类调用它的公有方法的事情吗(第43篇中提到过)?”
小白:“哦!想起来了,类不用直接使用它的公有方法,必须实例化才能用。所以如果让OldMan的原型prototype指向一个Man的实例就能复制它的所有属性和方法了,这个方法真是太妙了。稍等我试试啊!”
小白折腾了半天,郁闷的跟老朱说道:“不行啊!我让OldMan的原型等于Man的实例,新创建一个OldMan的实例以后输出name居然是undefined未定义。”
老朱郁闷的说道:“为啥把我定义成老男人了?”
小白:“哈哈,你头发都快成银白色了,还不是老男人啊!朱哥,为啥我这里会出错呢?”
老朱:“你有时间了解一下什么是构造函数,我这里就不多说了,一般我们实例化一个类的时候会通过new关键字,如果一个类的构造函数需要传入参数,我们在new这个类的时候也必须传参。你再看看代码中Man的function中有name和tel两个参数,而你定义的OldMan类构造函数是没有参数的。”
小白:“恩,确实是这样。”
老朱:“所以你犯了两个错,一个是OldMan.prototype = new Man()这里没有传参,一个是‘varlaozhu= newOldMan("老朱","8888888");’这里传了参数,OldMan的原型指向Man的实例可不能替换OldMan的构造函数,这个你能明白么?”
小白:“有点晕,我还得消化消化,那应该怎么改呢?”
老朱:“如果把父类中构造函数的两个参数去掉,赋值改为通过公共方法赋值你理解起来就容易多了。”
小白:“哦!这个我能看懂,您现在定义了三个私有变量,更改和获取这三个变量通过公共方法来实现,这样就能控制三个私有变量的读写功能了。我再试试继承”
小白激动的说道:“哇,可以了,完美的继承了Man的公共方法。”
老朱跟小白说:“我们可以通过instanceof检测某一个对象是否属于某个类。现在我们看看实例laozhu的所属。”
“怎么样?你再想想laozhu是不是属于Object类。”
小白说道:“当然属于了,Object是顶级类啊。”
“恩,不错!不过,今天我们在做继承的时候碰到了构造函数的问题,所以现在做的这个继承还不是完美的继承。另外你考虑一下,我们把类的原型指向了父类的实例,虽然可以把方法和属性继承过来,可是碰到父类的属性中有复杂数据类型(数组、对象等都是引用类型,成长之路7中提到过引用类型)的时候,就会出现问题,这些我们后面继续讨论。你先把今天说的这些消化消化吧!”
想学H5的朋友可以关注老炉,您的关注是我持续更新《小白HTML5成长之路》的动力!