理解javascript函数的关键就是抛弃主流面向对象语言中“类”的观念。
主流的面向对象语言,例如C++,先写好“类”的定义。当需要对象时,用new关键字去划出一片内存来,赋上初值,就有了一个对象(“类”的“实例”)。混淆的来源是javascript语言模仿了面向对象的语法,看到Object,new之类的关键字,很难不让人去联想那些传统的面向对象的经典概念。
我们可以优雅的模仿“类”的观念,但是始终要提醒自己,javascript只有对象,没有类。
- //Animal是一个对象
- Animal = {
- createNew: function( bundle ) {
- var animal = {};
- var protect = bundle || {};//bundle传递的是指针,修改protect时外面对象会跟着变(除非不传)
- protect.sound = 'growl';
- protect.makeSound = function(){
- return protect.sound;
- }
- return animal;
- }
- }
- //Cat也是一个对象
- Cat = {
- createNew: function(mySound) {
- var protect = {};
- var cat = Animal.createNew( protect );//protect会被修改,然后返回空白对象{}给cat
- protect.sound = mySound;
- cat.meow = function(){ return protect.makeSound(); };//cat要调用protect中的方法
- return cat;
- }
- }
- //javascript区分大小写,通过Cat对象构造一个新的对象赋值给cat
- var cat = Cat.createNew("meow!");
- pt("cat.sound");//cat不能直接访问sound
- pt("cat.meow()");//通过函数可以访问sound
- var bigCat = Cat.createNew("meow!meow!meow!");
- pt("bigCat.sound");//bigCat也不能直接访问sound
- pt("bigCat.meow()");//通过函数可以访问sound
调试信息:
cat.sound undefined
cat.meow() meow!
bigCat.sound undefined
bigCat.meow() meow!meow!meow!
[备注] 如果希望大猫(bigCat)小猫(cat)有公共的属性,因为Cat本身也是一个对象,所以可以在Cat中定义对象作为公共属性,并在createNew中增加操纵此公共属性的函数。
模仿“类”的观念让我联想到了东施效颦的典故。东施模仿西施优雅的举止肯定是不能达到西施的水准,但是东施难道就没有自己的特长么?例如,做家务会不会比西施更能干呢?javascript函数最大的好处就是可以赋值给变量,因而我们可以编写“函数的函数”,或者,把函数当参数传来传去,函数也可以有自己的方法,诸如此类。还记得在学习早期结构化的编程语言(例如Fortran和C)时,被纠正说不能传递函数当参数么?
- Function.prototype.run=function(){
- return "run~~";
- }
- pt("cat.meow.run()");//函数也可以有方法
调试信息:
cat.meow.run() run~~
下面玩一个克隆函数的小把戏:
- //有此方法今后不需要再写prototype关键字
- Function.prototype.method = function(name,func){
- this.prototype[name] = func;
- return this;
- };
- //添加cloneFunction方法
- Function.method("cloneFunction",function(){
- var slice = Array.prototype.slice;
- var args = slice.apply(arguments);
- var that = this;
- return function(){
- return that.apply(null,args.concat(slice.apply(arguments)));//此处slice.apply(arguments)含义与前处不同
- }});
- //定义add函数
- var add = function(a,n){
- return a+n;
- }
- pt("add(0,5)");//调试输出
- var add1 = add.cloneFunction(1);//克隆1
- var add2 = add.cloneFunction(2);//克隆2
- var add3 = add.cloneFunction(3);//克隆3
- pt("add1(5)");//调试输出
- pt("add2(5)");//调试输出
- pt("add3(5)");//调试输出
调试信息:
add(0,5) 5
add1(5) 6
add2(5) 7
add3(5) 8
想获得全部可执行代码,请下载附件。
附件:http://down.51cto.com/data/2362039
本文转自 hexiaini235 51CTO博客,原文链接:http://blog.51cto.com/idata/1101621,如需转载请自行联系原作者