介绍

函数对于任何语言都是一个核心的概念,可以封装任意多条语句。

ECMAScript函数不介意传递多少函数,什么类型,因为在函数内部都是用一个数组来表示,你可以通过访问函数内部arguments对象来访问。

函数重载

传统意义上的重载,比如说在java中,可以为一个函数编写两个定义,只要接受的参数类型和数量(签名)不一样即可,但是ECMAScript参数实际上是一个数组来表

示的,所以没有签名概念,继而也不会有纯正的重载。

但是可以通过判断arugumens.length来实现重载。

function test(){
  if(arguments.length == 1){...}else{...}
}

JS中的函数可以定义成

var test = function(){}

如果定义两个test函数,之后定义的函数会覆盖之前的定义,并不会实现重载。

arguments的内部属性

函数有个length属性,是函数定义时的参数的个数;相对于function.length,函数内部的arguments.length更加的动态,是函数执行时传进去的参数,有可能和

function.length不同。

arguments.callee

arguments对象有个属性(也是个对象)callee ,指向拥有这个arguments对象的函数。

阶乘函数

Function 入参可以是多个吗_数组

Function 入参可以是多个吗_数组_02

View Code

1 function factorial(){    
2 if(num <= 1){
3 return 1;
4 }else{
5 return num*(factorial(num-1));
6 }
7 }

 其中函数的执行与函数名紧密耦合了,也就是说改动函数名的话,代码也需要改动,为了消除这种耦合,可以稍作改动

function factorial(num){
        if(num <= 1){
            return 1;
        }else{
            return num*(argumens.callee(num-1));
        }
}

* 对这个编辑器还不了解,不知道怎么修改写上去的代码,上面的括号应该改一下,还有argumens改为arguments;

arguments.callee.caller

可以用来追踪调用arguments.callee函数的函数。

函数柯里化

函数柯里化是把接受多个参数的函数变成接受单一参数的函数,我的理解就是动态成单一参数的函数,柯里化的过程就是生产新函数的过程。

比如说我要局部刷新页面,我在哪里(参数1)放入数据(参数2),这个时候需要两个参数。

但是可以通过柯里化实现调用单一参数的函数。

function update(item,W){
    return function(text){
        $("div#"+item).html(text + ' ' + W);
    }
}

$.ajax({
    ...
    success:function(data){
        update('divID','XOX')(data);//update('divID','XOX')就是一个柯里化的过程
    }
});

求值策略(按共享传递)

按值传递

按值传递,很多开发人员都很了解了,参数的值是调用者传递的对象值的拷贝(copy of value),函数内部改变参数的值不会影响到外面的对象(该参数在外面的值),一般来说,是重新分配了新内存(我们不关注分配内存是怎么实现的——也是 是栈也许是动态内存分配),该新内存块的值是外部对象的拷贝,并且它的值是用到函数内部的。

按引用传递

另外一个众所周知的按引用传递接收的不是值拷贝,而是对象的隐式引用,如该对象在外部的直接引用地址。函数内部对参数的任何改变都是影响该对象在函数外部的值,因为两者引用的是同一个对象,也就是说:这时候参数就相当于外部对象的一个别名。

按共享传递

函数内部给参数重新赋新值不会影响到外部的对象,但是因为该参数是一个地址拷贝,所以在外面访问和里面访问的都是同一个对象,改变该参数对象的属性值将会影响到外部的对象。

一般理论确实有按值传递的说法:但这时候这个值就是我们所说的地址拷贝(副本),因此并没有破坏规则。

简单的说,参数如果是普通类型,那么是按值传递;

如果是传递的是引用类型,对该参数(地址,指向函数外的对象)做局部修改,会影响到函数外的对象,如果对该参数(局部变量,而且是个地址)重新赋值,智慧让该参数指向一个新的对象,函数执行完之后就会销毁。