什么是函数?
函数实际上是对象,每个函数都是Function类型的实例,都和其他引用类型一样具有属性和方法。“函数是对象,函数名是指针”。要访问函数的指针而不是执行函数的话,必须去掉函数名后面的()
定义函数的方式
解析器会率先读取函数声明,并使其在执行任何代码之前可以访问。
函数表达式必须等到解析器执行到它所在行时才会被解释执行。
1、函数声明
组成格式:function关键字+函数名字
许多浏览器定义了一个非标准的name属性,通过这个属性可以访问到函数给定的名字
function sayFan(){
console.log(111)
}
console.log(sayFan.name) //sayFan
函数声明提升
sayFan()
function sayFan(){
console.log(111)
}
在执行代码前会先读取函数声明,意味着可以把函数声明放到调用它的语句的后面
2、函数表达式
匿名函数(拉姆达函数)
创建一个函数并将它赋值给变量,function关键字后面没有标识符
name属性是空字符串
没有函数声明提升,使用前必须先赋值
var sayFan=function(arg0,arg1,arg2){
//函数体
};
3、Function构造函数
可以接收任意数量的参数,最后一个参数被认为是函数体
var sum=new Function("console.log(111)")
console.log(sum)
函数没有重载
当声明两个同名函数时,后面的函数会覆盖前面的函数
函数内部属性
在函数内部,有两个特殊的对象:arguments和this。
arguments是一个类数组对象,包含所有传进函数中的参数。他的主要用途是保存函数参数,但是他有一个callee属性。该属性是一个指针,指向拥有arguments对象的函数。
function factorial(num){
if(num<=1){
return 1;
}else {
//return num*factorial(num-1) 等价 为了消除函数的执行和函数名紧密耦合的现象
return num*arguments.callee(num-1)
}
}
this引用的是函数的执行环境。caller属性保存着调用当前函数的函数的引用,在全局作用域中值为null。
在严格模式下arguments.callee和arguments.caller会导致错误,非严格模式下arguments.caller属性的值为undefined
函数的属性和方法
函数是对象,所以有属性和方法。
每个函数都有这两个属性:length和prototype。
- length属性:函数接收参数的个数
- prototype属性:保存所有实例的方法,例如toString()、valueOf()等方法都保存在prototype下。在ES5中,property是不可枚举的。
每个函数都有两个非继承而来的方法:apply()和call(),设置函数体内this的值。他们真正强大的用处是扩充函数的作用域
递归函数
递归函数是在一个函数通过名字调用自身的情况下构成的
闭包
闭包是指有权访问另外一个函数作用域中的变量的函数。创建一个闭包函数即在一个函数内部创建另外一个函数