注册 登录

清河洛

javascript中的函数

qingheluo2016-12-16清河洛294
函数对任何语言来说都是一个核心的概念。通过函数可以封装任意多条语句,而且可以在任何地方、任何时候调用执行。在ECMAScript中,Function(函数)类型实际上是对象。每个函数都是Function类型的实例,而且都与其他引用类型一样具有属性和方法。由于函数是对象,因此函数名实际上也是一个指向函数对象的指针。一.函数的声明方式1.普通的函数声明function box(num1,num2){return num1+ num2;}2.使用变量初始化函数var box= function(num1,num2){return num1 + num2;};3.使用Function构造函数va...

函数对任何语言来说都是一个核心的概念。通过函数可以封装任意多条语句,而且可以在任何地方、任何时候调用执行。在ECMAScript中,Function(函数)类型实际上是对象。每个函数都是Function类型的实例,而且都与其他引用类型一样具有属性和方法。由于函数是对象,因此函数名实际上也是一个指向函数对象的指针。

一.函数的声明方式

1.普通的函数声明

function box(num1,num2){return num1+ num2;}

2.使用变量初始化函数

var box= function(num1,num2){return num1 + num2;};

3.使用Function构造函数

var box= new Function('num1','num2','return num1 + num2');

第三种方式我们不推荐,因为这种语法会导致解析两次代码(第一次解析常规ECMAScript代码,第二次是解析传入构造函数中的字符串),从而影响性能。但我们可以通过这种语法来理解"函数是对象,函数名是指针"的概念。

ECMAScript中的函数,重复声明名称相同的函数会执行最后一次声明的函数。

ECMAScript中的函数是对象,因此函数也有属性。每个函数都包含length属性,表示函数希望接收的命名参数的个数。

二.return返回值

带参和不带参的函数,都没有定义返回值,而是调用后直接执行的。实际上,任何函数都可以通过return语句跟后面的要返回的值来实现返回值。

return语句还有一个功能就是退出当前函数,注意和break的区别(break用在循环和switch分支语句里)。

三.arguments对象

ECMAScript函数不介意传递进来多少参数,也不会因为参数不统一而错误。函数体内可以通过arguments对象来接收传递进来的参数。arguments会以数组的形式保存函数的传入参数,如arguments[0]表示第一个传入的参数,arguments[1]表示第二个,以此类推。arguments对象的length属性可以得到参数的数量,利用length这个属性,来智能的判断有多少参数,然后把参数进行合理的应用。

arguments对象还有一个名叫callee的属性,该属性是一个指针,指向拥有这个arguments对象的函数:

function box(num){

if (num <= 1){return 1;}

else {return num * box(num-1);}

}

上例中阶乘函数用到了递归算法,所以函数内部一定会调用自身;如果函数名不改变是没有问题的,但一旦改变函数名,内部的自身调用需要逐一修改。为了解决这个问题,我们可以使用arguments.callee来代替。

function box(num) {

if (num <= 1) {return 1;}

else {return num * arguments.callee(num-1);}//使用callee来执行自身

}

四、rest 参数

ES6引入rest参数(形式为...变量名),用于获取函数的多余参数,这样就不需要使用arguments对象了。

rest参数变量是一个数组,该变量将多余的参数放入数组中。

function add(...values) {
  let sum = 0;
  for (var val of values) {
    sum += val;
  }
  return sum;
}
上例中变量values就表示所有的传入参数组成的数组

五、JS中的函数执行顺序

JS中声明的函数只有在调用时才会运行,否则不会运行(这是句废话)

如果函数中有语法错误,只要没有调用,就不会报错

function demo(a,b,c){
    console.log(a,b,c);
    var a = 'a';
    var b = function (){};
    (function a(){});
    function c(){}
    console.log(a,b,c);
}
demo(1,2,3);

当函数被调用了以后,会在内存中为该函数创建一个内存空间,来记录这个函数中用到的信息,如变量、函数声明等,我们把这个内存空间称为函数执行上下文

创建函数执行上下文之的顺序

一、参数填充

对函数中的参数进行赋值操作,此时
a=1
b=2
c=3

二、查找函数声明

在函数中查找是否有函数声明语句,如果有则解析
在以上代码中的function c(){}为一个函数声明
而var b = function(){}语句虽然值也是函数,但是输于赋值操作,后面的部分称为函数表达式,并非函数声明
(function a(){});语句因为添加了括号,所以也是一个函数表达式而非函数声明
解析该语句后如果变量中有重名则覆盖,所以此时会覆盖变量c,此时
    a=1
    b=2
    c=function c(){}

三、查找变量声明

在函数中查找变量声明,并为这些变量赋值为undefinde
如果有变量重名,则忽略该变量声明
上例中有2个变量声明语句,var a = 'a';和var b = function (){};
但是检测已存在变量a和b,所以忽略这两个声明变量语句,此时
    a=1
    b=2
    c=function c(){}

至此函数执行上下文创建完毕,开始执行函数的内容

第一行的console.log(a,b,c);语句执行结果打印

1 2 ƒ c(){}

后面执行两个赋值操作语句,执行完后函数执行上下文中:

a='a'
b=function (){}
c=function c(){}

(function a(){});语句不产生任何作用

function c(){}语句在创建函数执行上下文时已经解析,所以跳过

执行最后的console.log(a,b,c);语句,打印

a ƒ (){} ƒ c(){}

至此,函数执行完毕



网址导航