详细的JavaScript基础语法-中-函数

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接: https://blog.csdn.net/qq_39206750/article/details/102540960

函数

函数就是封装了一段可被重复调用执行的代码块。通过此代码块可以实现大量代码的重复使用
函数就是程序代码的封装

		// 声明函数
		function 函数名() {
		    //函数体代码
		}
		// 调用函数
        函数名();  // 通过调用函数名来执行函数体代码
       
	         //1、声明函数
	      function hello(){
	          console.log('hi~~');
	      }
	      //2、调用函数 
	     console.log( hello());//如果函数代码体为空则为Undefind
  • function 是声明函数的关键字,必须小写
  • 由于函数一般是为了实现某个功能才定义的, 所以通常我们将函数名命名为动词,比如 getSum
  • 调用的时候千万不要忘记添加小括号
  • 口诀:函数不调用,自己不执行
   注意:声明函数本身并不会执行代码,只有调用函数时才会执行函数体代码。

函数的参数
参数语法

  • 形参:函数定义时设置接收调用时传入
  • 实参:函数调用时传入小括号内的真实数据
    参数语法
	// 带参数的函数声明
	// 可以定义任意多的参数,用逗号分隔
	function 函数名(形参1, 形参2 , 形参3...) { 
	  // 函数体
	}
	// 带参数的函数调用
	函数名(实参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,所以无法正确调用

猜你喜欢

转载自blog.csdn.net/qq_39206750/article/details/102540960