函数浅析

首先我们要思考,为什么需要函数这个东西。函数的出现涉及一个编程原则:高内聚 低耦合。

低耦合是指降低代码的重复率,不要写冗余的代码。高内聚是指代码紧密联系性强,独立性强。使得代码能够实现模块的单一责任制,一个模块负责一个功能,各司其职,然后组合在一起。

当代码耦合性过高时,就需要解耦合,JS中解耦的常用方法就是抽离成函数,定义函数,是一个固定的功能或者程序段被封装的过程。实现这个过程需要在这个封装体中设置一个入口和一个出口,入口就是参数,出口就是返回(值)。

定义函数

定义函数的方式有两种:

函数声明 该方式定义的函数,函数名变成了一个变量,这个变量的值就是函数本身。

scss 复制代码
function test(){
    console.log('HelloWorld');
}
test(); // 'HelloWorld'

函数表达式 看起来像函数声明,但它的函数名是可选的,大部分情况下是不写的(即匿名函数)。除非函数执行过程中需要引用自身,比如阶乘函数。如果函数表达式包含名字,则该函数的局部作用域中会包含一个该名字与函数对象的绑定。

javascript 复制代码
const test = function(){
    console.log('HelloWorld');
}
test(); // 'HelloWorld'

函数属性

函数是一种特殊对象,它也有自己的属性,其中有些常用的属性。

name属性 可以获得函数的名字。

javascript 复制代码
function test(){
    console.log('HelloWorld');
}
console.log(test.name); // 'test'
javascript 复制代码
const test = function(){
    console.log('HelloWorld');
}
console.log(test.name); // 'test'

length属性 可以获得函数形参的数量。

javascript 复制代码
function test(a, b, c){
    console.log(a, b, c);
}
console.log(test.length); // 3

函数形参与实参

形参与默认值

调用函数时,如果没有给形参设定默认值,并且也没有传入实参,那么参数的默认值是undefined

scss 复制代码
function test(a, b){
    console.log(a); // 1
    console.log(b); // undefined
}

test(1);

如果需要给参数指定默认值,可以在定义函数时给形参设定默认值,注意这种方式是ES6的写法。

scss 复制代码
function test(a = 1, b = 2){
    console.log(a); // 1
    console.log(b); // 2
}

test();

如果形参设定了默认值,实参也进行了传值,那么就需要分为多种情况:

  • 第一种是形参默认值是undefined,实参是有效值。这种和形参不设定默认值的效果是一样的,输出结果是实参的值。
scss 复制代码
function test(a = undefined, b){
    console.log(a);
    console.log(b);
}

test(1, 2); // 1 2
  • 第二种是形参默认值是有效值,实参是undefined。此时会选择形参的默认值使用,这种形式通常是为了方便给后面的形参传值。
scss 复制代码
function test(a = 1, b){
    console.log(a);
    console.log(b);
}

test(undefined, 2);  // 1 2
  • 第三种就不用说了,形参和实参都是undefined,那结果自然是undefined

arguments对象

ES5中,并没有直接给形参设定默认值的写法,所以如果需要设定默认值,需要使用arguments对象,它是所有(非箭头)函数中都可用的局部变量,此对象包含传递给函数的每个实参,可以通过它在函数中引用函数的实参,它是一个类数组,第一个实参位于索引0处。

css 复制代码
function test(a, b){
// 这种方式也是变相的实现了为参数设定默认值
    var a = arguments[0] || 1;
    var b = arguments[1] || 2;
    console.log(a);
    console.log(b);
}

test(3); // 3 2

函数递归

函数递归就是函数自己调用自己,比如不用for循环,实现n的阶乘。要注意,递归一定要有一个出口,不然就会无限递归下去,比如以下代码中fact(1)就是出口,因为当n为1时,会返回一个确定值,这样就可以开始计算之前排队等着的运算。

scss 复制代码
function fact(n){
    if(n === 1){
        return 1;
    }else{
        return n * fact(n - 1);
    }
}
console.log(fact(5)); // 120

函数递归实现斐波那契数列。

scss 复制代码
// 规律:n3 = n2 + n1;
// 出口:n <= 0    n <= 2;
function fb(n){
    if(n <= 2){
        if(n <= 0){
            return 0;
        }else{
            return 1;
        }
    }else{
        return fb(n - 1) + fb(n - 2);
    }
}
console.log(fb(10)); // 55
相关推荐
Hilaku37 分钟前
我用 Gemini 3 Pro 手搓了一个并发邮件群发神器(附源码)
前端·javascript·github
IT_陈寒37 分钟前
Java性能调优实战:5个被低估却提升30%效率的JVM参数
前端·人工智能·后端
快手技术38 分钟前
AAAI 2026|全面发力!快手斩获 3 篇 Oral,12 篇论文入选!
前端·后端·算法
颜酱40 分钟前
前端算法必备:滑动窗口从入门到很熟练(最长/最短/计数三大类型)
前端·后端·算法
全栈前端老曹1 小时前
【包管理】npm init 项目名后底层发生了什么的完整逻辑
前端·javascript·npm·node.js·json·包管理·底层原理
HHHHHY1 小时前
mathjs简单实现一个数学计算公式及校验组件
前端·javascript·vue.js
boooooooom1 小时前
Vue3 provide/inject 跨层级通信:最佳实践与避坑指南
前端·vue.js
一颗烂土豆1 小时前
Vue 3 + Three.js 打造轻量级 3D 图表库 —— chart3
前端·vue.js·数据可视化
青莲8431 小时前
Android 动画机制完整详解
android·前端·面试
iReachers1 小时前
HTML打包APK(安卓APP)中下载功能常见问题和详细介绍
前端·javascript·html·html打包apk·网页打包app·下载功能