函数
函数就是封装了一段可被重复调用执行的代码块。通过此代码块可以实现大量代码的重复使用
函数就是程序代码的封装
;
// 声明函数
function 函数名() {
//函数体代码
}
// 调用函数
函数名(); // 通过调用函数名来执行函数体代码
//1、声明函数
function hello(){
console.log('hi~~');
}
//2、调用函数
console.log( hello());//如果函数代码体为空则为Undefind
- function 是声明函数的关键字,必须小写
- 由于函数一般是为了实现某个功能才定义的, 所以通常我们将函数名命名为动词,比如 getSum
- 调用的时候千万不要忘记添加小括号
- 口诀:函数不调用,自己不执行
注意:声明函数本身并不会执行代码,只有调用函数时才会执行函数体代码。
函数的参数
参数语法
- 形参:函数定义时设置接收调用时传入
- 实参:函数调用时传入小括号内的真实数据
// 带参数的函数声明
// 可以定义任意多的参数,用逗号分隔
function 函数名(形参1, 形参2 , 形参3...) {
// 函数体
}
// 带参数的函数调用
函数名(实参1, 实参2, 实参3...);
- 调用的时候实参值是传递给形参的
- 形参简单理解为:不用声明的变量
- 实参和形参的多个参数之间用逗号(,)分隔
- 当函数参数不匹配的时候
function getSum(num1, num2) {
console.log(num1 + num2);
}
//实参与形参个数一致、正常输出
getSum(1, 2);//3
//实参的个数多于形参、取形参的个数
getSum(1, 2, 3);//3
//实参的个数少于形参、形参就是undefined、
//形参可以看成是一个没有赋值的变量、num2没有给到值就变成undefind了
getSum(1);//NaN 1+undefind=NaN
注意:在JavaScript中,形参的默认值是undefined。
- 函数可以带参数也可以不带参数
- 声明函数的时候,函数名括号里面的是形参,形参的默认值为 undefined
- 调用函数的时候,函数名括号里面的是实参
- 多个参数中间用逗号分隔
- 形参的个数可以和实参个数不匹配,但是结果不可预计,我们尽量要匹配
- 我们利用函数的参数实现函数重复不同的代码
函数返回值
- 返回值:函数调用整体代表的数据;函数执行完成后可以通过return语句将指定数据返回 。
// 声明函数
function 函数名(){
...
return 需要返回的值;
}
// 调用函数
函数名(); // 此时调用函数就可以得到函数体内return 后面的值
function getSum(num1,num2) {
//return的返回值会返回给调用者(赋值给getSum())
//return后面的结果=函数名()
return num1+num2;
}
console.log(getSum(1,2));
- 在使用 return 语句时,函数会停止执行,并返回指定的值
- 如果函数没有 return ,返回的值是 undefined
break ,continue ,return 的区别
- break :
结束当前的循环体
(如 for、while)- continue :
跳出本次循环,继续执行下次循环
(如 for、while)- return :
不仅可以退出循环,还能够返回 return 语句中的值,
同时还可以结束当前的函数体内的代码
arguments的使用
arguments是一个伪数组,可以进行遍历。伪数组具有以下特点
- 具有 length 属性
- 按索引方式储存数据
- 不具有数组的 push , pop 等方法
注意:在函数内部使用该对象,用此对象获取函数调用时传的实参。
// arguments 的使用 只有函数才有 arguments对象
//而且是每个函数都内置好了这个arguments
function fn() {
// 里面存储了所有传递过来的实参 arguments = [1,2,3]
// console.log(arguments);
// console.log(arguments.length);
// console.log(arguments[2]);
// 我们可以按照数组的方式遍历arguments
for (var i = 0; i < arguments.length; i++) {
console.log(arguments[i]);
}
}
fn(1, 2, 3);
fn(1, 2, 3, 4, 5);
// 伪数组 并不是真正意义上的数组
// 1. 具有数组的 length 属性
// 2. 按照索引的方式进行存储的
// 3. 它没有真正数组的一些方法 pop() push() 等等
函数案例
函数内部可以调用另一个函数,在同一作用域代码中,函数名即代表封装的操作,使用函数名加括号即可以将封装的操作执行。
//判断是平年还是闰年
isRunYear = function (year) {
var flag = false;
if (year % 4 == 0 && year % 100 !== 0 || year % 400 == 0) {
flag = true;
}
return flag;
}
function backDay() {
var year = prompt('请输入你的年份');
//在此处把函数放到判断中、isRunYear(year)会返回一个false/true、
//然后if进行判断
//匿名函数的调用函数、必须写到函数体后面
if (isRunYear(year)) {
alert('闰年有29天')
} else {
alert('平年有28天')
}
}
backDay()
函数的声明方式
- 自定义函数方式(命名函数)
利用函数关键字 function 自定义函数方式、
// 声明定义方式
function fn() {...}
// 调用
fn();
命名函数的调用函数的代码既可以放到声明函数的前面,也可以放在声明函数的后面
- 函数表达式方式(匿名函数)
-利用函数表达式方式的写法:
// 这是函数表达式写法,匿名函数后面跟分号结束
var fn = function(){...};
// 调用的方式,函数调用必须写到函数体下面
fn();
- 因为函数没有名字,所以也被称为匿名函数
- 这个fn 里面存储的是一个函数
- 函数表达式方式原理跟声明变量方式是一致的
函数调用的代码必须写到函数体后面
作用域
- 全局作用域和局部作用域:
1、全局作用域就是在整个程序中都可以使用的东西
2、作用于函数内的代码环境就是局部作用域、所以也称为函数作用域。
- 全局变量和局部变量;
1、全局变量就是定义在全局作用域中的变量;
2、局部变量就是定义在局部作用域中的变量(定义在函数内的变量)
- 作用域链:
只要是代码都一个作用域中,写在函数内部的局部作用域,未写在任何函数内部即在全局作用域中;如果函数中还有函数,那么在这个作用域中就又可以诞生一个作用域;根据在**[
内部函数可以访问外部函数变量
]**的这种机制,用链式查找决定哪些数据能被内部函数访问,就称作作用域链
-查找一个变量的方式顺序。采取就近原则。
// 作用域链:内部函数访问外部函数的变量,采取的是链式查找的方式来决定取那个值
// 这种结构我们称为作用域链 就近原则
var num = 10;
function fn() { // 外部函数
var num = 20;
function fun() { // 内部函数
console.log(num);//根据就近原则至上查找num
}
fun();
}
fn();
-作用域案例
// 结果是几?
var a = 1;
function fn1() {
var a = 2;
var b = '22';
fn2();
function fn2() {
var a = 3;
fn3();
function fn3() {
var a = 4;
console.log(a); //a的值 ?//
console.log(b); //b的值 ?//
}
}
}
fn1();
//根据作用域链依次往上寻找、a为4、b ='22'
作用域链案例解析图
预解析(变量、函数提升)
- 预解析相关概念
**JavaScript 代码是由浏览器中的 JavaScript 解析器来执行的。
**JavaScript 解析器在运行 JavaScript 代码的时候分为两步:预解析和代码执行。
- 预解析:
**在当前作用域下, JS 代码执行之前,
**浏览器默认把带有 var 和 function 声明的变量在内存中进行提前声明或者定义;
- 代码执行
**从上到下执行JS语句
预解析会把变量和函数的声明在代码执行之前执行完成。
变量预解析
- 变量提升(变量预解析):
**变量的声明会被提升到当前作用域的最上面,`变量的赋值不会提升。
console.log(num); // 结果是多少?
var num = 10;
//结果:undefined
注意:变量提升只提升声明,不提升赋值
函数预解析
函数提升: ** 函数的声明会被提升到当前作用域的最上面,但是不会调用函数。
fn();
//函数提升会默认把什么函数放到调用函数之前
function fn() {
console.log('打印');//输出打印
}
注意:函数声明代表函数整体,所以函数提升后,函数名代表整个函数,但是
函数并没有被调用
!
函数表达式(匿名函数)声明函数问题
fn();
var fn = function() {
console.log('想不到吧');
}
//结果:报错提示 ”fn is not a function"
//此时的相当与一个未赋值的变量、
//执行匿名函数一定要先把声明函数放前面
解释:该段代码执行之前,会做变量声明提升,fn在提升之后的值是undefined;而
fn调用是在fn被赋值为函数体之前
,此时fn的值是undefined,所以无法正确调用