javascript 函数的回调和递归

前言

先来看看我们为什么要使用函数:

函数对任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且可以在任何地方、任何时候调用执行。在javascript里,函数即对象,程序可以随意操控它们。函数可以嵌套在其他函数中定义,这样它们就可以访问它们被定义时所处的作用域中的任何变量,它给javascript带来了非常强劲的编程能力。

函数的声明:

1、命名函数。

通常来说函数的声明会有一个返回值,如果没有设置return,该函数默认返回未undefined,如果设置,将返回该值。执行函数后会返回该值。

function  abc(){
	var s=3+4return s;
}
var s1=abc();
s1===7;

2、匿名函数,声明函数的时候没有赋予名称。

函数声明好以后赋值给了box,这里注意box并不是num1+num2的值,而是function定义的这个匿名函数。

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

在对象中,可以设置一个属性是一个函数,这样就给了对象定义了一个方法。

var obj={
	name:”zhangsan”,
	fire:function(num1,num2){
	    var s=num1+num2;
		return s;
	}
}

3、构造函数,使用Function构造函数。

var box=new Function('num1','num2','retun num1+num2');
//num1,num2参数,最后的字符串是函数体

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

函数的本质

JS中的函数是对象,因此函数也有属性和方法。每个函数都包含两个属性:length 和 prototype。其中,length 属性表示函数希望接收的命名参数的个数。

function fun1(a,b){
	let sum=a+b;
}
console.dir(fun1);//打印结果如下

在这里插入图片描述
函数是对象,因此也可以为函数定义属性和方法。

function fun1(){
	console.log(this);//window
}
fun1.a=3;
fun1.fun2=function(){
	//因为fun1是对象,因此,在执行fun2后,里面的this就是fun1
	console.log(this.a);//a
}
//执行函数,用fun1()
fun1();
//调用函数的方法或者属性时,就不能加小括号
fun1.fun2();

函数的执行

1、普通执行:函数名(),执行后,可以完成函数的代码内容,如果函数内有return 值,这时候执行后会返回该值。如何没有return,则执行完所有代码。

function abc() {
   return 3+4;
}
var s=abc();

2、函数的独立执行,函数自身是可以独立执行的,并且也可以把独立执行的结果赋值给一个变量。
注意,除了匿名函数可以独立执行,实名函数也是可以独立执行的,不过这种独立执行就不能在该函数中添加参数了,并且匿名函数,也只能在此执行一次,不能多次调用,显而易见的好处是,该函数内的变量统统是私有变量。

//匿名函数自己执行
var  s=0
(function(){
	s=4+5;
})()

//匿名函数赋值
var s=(function(){
	return 4+5;
})();

3、函数的call,语法:函数名.call(thisObj,arg1,arg2…)

call 方法可以用来代替另一个对象调用一个方法。

4、函数的apply,语法:apply(thisObj,[argArray])

如果 argArray 不是一个有效的数组或者不是 arguments 对象,那么将导致一个 TypeError。其实apply和call基本相似,这里传参的模式产生变化而已,在call中传参使用了逐个传入,apply中是以数组传参带入。

点击查看 call和apply 的详细内容:https://blog.csdn.net/Charissa2017/article/details/104368990

函数的删除

当函数不在使用时,就需要删除,函数也是对象,如果不删除,它将常驻内存中,如果该函数不再使用就可以使用删除彻底清除掉该函数。但是只有函数是匿名定义的或者通过构造函数创建的才可以被删除。

//匿名函数的删除:
var abc=function(){
    console.log("aaa");
}
abc();
abc=null;

//对象下的方法删除
var obj={
    abc:function(){
        console.log("bbb");
    }
}
obj.abc();
obj.abc=null;

函数的参数

  • 参数是指由外部传入到函数中的变量,仅作为变量使用,但是该变量可以是任何内容,包括函数。
  • 被传入的参数作为私有变量使用,可以被覆盖掉。
  • 外部传入的参数可以节省全局变量的定义,甚至保证函数中的部分变量的独立性。
  • ECMAScript 函数不介意传递进来多少参数,也不会因为参数不统一而错误,函数体内可以通过arguments 对象来接收传递进来的参数。

点击查看函数参数及arguments的详细内容:https://blog.csdn.net/Charissa2017/article/details/103742144

回调

回调函数,或简称回调,是指通过函数参数传递到其它代码的,某一块可执行代码的引用。这一设计允许了底层代码调用在高层定义的子程序。

回调的特征:

  • 在一个函数中执行另外一个函数,并且这个另外的函数必须以参数方式传入的;
  • 函数中不关心回调函数做了什么,也不关心回调函数是谁,只需要在需要执行的时候执行它;
  • 回调函数中 this 的指向将会重定向到window;
function fn(fn1){
	fn1(); 
}

fn(function(){
    console.log("aa");
})

递归

递归函数即自调用函数,在函数体内部直接或间接地自己调用自己,即函数的嵌套调用是函数本身。

被调函数运行的代码虽是同一个函数的代码体,但由于调用点,调用时状态, 返回点的不同,可以看作是函数的一个副本,与调用函数的代码无关,所以函数的代码是独立的。被调函数运行的栈空间独立于调用函数的栈空间,所以与调用函数之间的数据也是无关的。函数之间靠参数传递和返回值来联系,函数看作为黑盒。

function fn(n){
	//终点
	if(...) return ...;
	//递归
	return fn(...)...;
}
  • 递归,当前函数内,满足条件时,调用当前函数自身,这就叫做递归;
  • 递归特征是先进后出;
  • 递归可以进行深度遍历;

递归有两种调用形式:直接递归调用和间接递归调用。

直接递归调用:

//求1-10数字的和
var sum=0;
function fn(x) {
    if(x<1) return 1;
    return fn(x-1)+x;
}
console.log(fn(10)); 

间接递归调用:

function fun1(x) {
    var z=fun2(x+1);
    console.log(z);
}
function fun2(a){
    if(a<5){
        fun1(a+1);
    }else{
        return a;
    }
}
fun1(3);

深度遍历:

var o = {
    a:{b:1},
    c:{d:2}
}
getProp(o);
function getProp(o) {
    for (var prop in o) {
        if (typeof o[prop] === "object") {
            console.log(prop);
            getProp(o[prop])
        } else {
            console.log(prop);
            console.log(o[prop]);
        }
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值