15-JavaScript高级③(函数进阶)

1. 函数的定义和调用

1.1 函数的定义方式

  • 函数声明方式 function 关键字 (命名函数)
function fn(){}
  • 函数表达式(匿名函数)
var fn = function(){}
  • new Function()
var fn = new Function('参数1','参数2'..., '函数体')
var f = new Function('a', 'b', 'console.log(a + b)');
f(1, 2);

所有函数都是 Function 的实例(对象)

## 1.2 函数的调用
/* 1. 普通函数 */
function fn() {
	console.log('人生的巅峰');
}
 fn(); 
/* 2. 对象的方法 */
var o = {
  sayHi: function() {
  	console.log('人生的巅峰');
  }
}
o.sayHi();
/* 3. 构造函数*/
function Star() {};
new Star();
/* 4. 绑定事件函数*/
 btn.onclick = function() {};   // 点击了按钮就可以调用这个函数
/* 5. 定时器函数*/
setInterval(function() {}, 1000);  这个函数是定时器自动1秒钟调用一次
/* 6. 立即执行函数(自调用函数)*/
(function() {
	console.log('人生的巅峰');
})();

2. this

2.1 函数内部的this指向

调用方式this指向
普通函数对象window
构造函数调用实例对象 原型对象里面的方法也指向实例对象
对象方法调用所属对象
事件绑定方法绑定事件对象
定时器函数window
立即执行函数window

2.2 改变函数内部 this 指向

2.2.1 call方法

fun.call(thisArg,arg1,arg2,...)
call()方法可以调用一个对象,并改变函数的 this 指向。
其参数1是修改后的this指向,参数2,参数3…使用逗号隔开连接。
多用于继承父类属性。

2.2.2 apply方法

fun.apply(thisArg,[argsArray])
apply()方法可以调用一个对象,并改变函数的 this 指向。
可以利用apply借助于内置对象求最大、最小值

var arr = [1, 66, 3, 99, 4];
//var max = Math.max.apply(null, arr);
var max = Math.max.apply(Math, arr);
var min = Math.min.apply(Math, arr);

2.2.3 bind方法

fun.bind(thisArg,arg1,arg2,...)
bind()方法不调用函数 ,但改变函数的 this 指向。返回由指定的this值和初始化参数改造的原函数拷贝。

var btns = document.querySelectorAll('button');
for (var i = 0; i < btns.length; i++) {
    btns[i].onclick = function() {
        this.disabled = true;
        setTimeout(function() {
            this.disabled = false;
        }.bind(this), 2000);
    }
}

2.2.4 call、apply、bind三者的异同

  • 共同点 : 都可以改变this指向
  • 不同点:
    • call 和 apply 会调用函数, 并且改变函数内部this指向.
    • call 和 apply传递的参数不一样,call传递参数使用逗号隔开,apply使用数组传递
    • bind 不会调用函数, 可以改变函数内部this指向.
  • 应用场景:
    • call 经常做继承.
    • apply经常跟数组有关系. 比如借助于数学对象实现数组最大值最小值
    • bind 不调用函数,但是还想改变this指向. 比如改变定时器内部的this指向.

3. 严格模式

3.1 什么是严格模式

JavaScript 除了提供正常模式外,还提供了严格模式(strict mode)。ES5 的严格模式是采用具有限制性 JavaScript变体的一种方式,即在严格的条件下运行 JS 代码。

严格模式在 IE10 以上版本的浏览器中才会被支持,旧版本浏览器中会被忽略。

严格模式对正常的 JavaScript 语义做了一些更改:
1.消除了 Javascript 语法的一些不合理、不严谨之处,减少了一些怪异行为。
2.消除代码运行的一些不安全之处,保证代码运行的安全。
3.提高编译器效率,增加运行速度。
4.禁用了在 ECMAScript 的未来版本中可能会定义的一些语法,为未来新版本的 Javascript 做好铺垫。比如一些保留字如:class,enum,export, extends, import, super 不能做变量名

3.2 开启严格模式

  • 为脚本开启严格模式:
<!-- 为整个脚本(script标签)开启严格模式 -->
<script>
    'use strict';
    //   下面的js 代码就会按照严格模式执行代码
</script>
<script>
    (function() {
        'use strict';
    })();
</script>
  • 为函数开启严格模式:
function fn(){
  'use strict';
} 

3.3 严格模式中的变化

  • 严格模式下,不允许使用未声明的变量
  • 严格模式下,不允许删除定义好的变量
  • 严格模式下,全局作用域中函数中的 this 是 undefined
  • 严格模式下,如果 构造函数不加new调用, this 指向的是undefined 如果给他赋值则 会报错
  • 严格模式下,定时器 this 还是指向 window
  • 严格模式下,函数里面的参数不允许有重名

更多严格模式要求参考

4. 高阶函数

高阶函数是对其他函数进行操作的函数,它接收函数作为参数将函数作为返回值输出

5. 闭包

5.1 变量的作用域复习

变量根据作用域的不同分为两种:全局变量局部变量

  1. 函数内部可以使用全局变量。
  2. 函数外部不可以使用局部变量。
  3. 当函数执行完毕,本作用域内的局部变量会销毁。

5.2 什么是闭包

闭包(closure)指有权访问另一个函数作用域中变量的函数。简单理解就是 ,一个作用域可以访问另外一个函数内部的局部变量。

// 闭包: fun 这个函数作用域 访问了另外一个函数 fn 里面的局部变量 num
//fn就是一个闭包
function fn() {
    var num = 10;

    function fun() {
        console.log(num);
    }
    fun();
}
fn();

5.3 闭包的作用

作用:延伸变量的作用范围。

 function fn() {
   var num = 10;
   function fun() {
       console.log(num);
 	}
    return fun;
 }
var f = fn();
f();

5.4 闭包的案例

  1. 利用闭包的方式得到当前li 的索引号
// 闭包应用-点击li输出当前li的索引号
// 1. 我们可以利用动态添加属性的方式
var lis = document.querySelector('.nav').querySelectorAll('li');
for (var i = 0; i < lis.length; i++) {
    lis[i].index = i;
    lis[i].onclick = function() {
        console.log(this.index);

    }
}
// 2. 利用闭包的方式得到当前小li 的索引号
for (var i = 0; i < lis.length; i++) {
    // 利用for循环创建了4个立即执行函数
    // 立即执行函数也成为小闭包因为立即执行函数里面的任何一个函数都可以使用它的i这变量
    (function(i) {
        // console.log(i);
        lis[i].onclick = function() {
            console.log(i);
        }
    })(i);
}
  1. 闭包应用-3秒钟之后,打印所有li元素的内容
 for (var i = 0; i < lis.length; i++) {
   (function(i) {
     setTimeout(function() {
     console.log(lis[i].innerHTML);
     }, 3000)
   })(i);
}
  1. 闭包应用-计算打车价格
/*需求分析
打车起步价13(3公里内),  之后每多一公里增加 5块钱.  用户输入公里数就可以计算打车价格
如果有拥堵情况,总价格多收取10块钱拥堵费*/

 var car = (function() {
     var start = 13; // 起步价  局部变量
     var total = 0; // 总价  局部变量
     return {
       // 正常的总价
       price: function(n) {
         if (n <= 3) {
           total = start;
         } else {
           total = start + (n - 3) * 5
         }
         return total;
       },
       // 拥堵之后的费用
       yd: function(flag) {
         return flag ? total + 10 : total;
       }
	}
 })();
console.log(car.price(5)); // 23
console.log(car.yd(true)); // 33

5.5 思考

var name = "The Window";
        var object = {
            name: "My Object",
            getNameFunc: function() {
                return function() {
                    return this.name;
                };
            }
        };

console.log(object.getNameFunc()());//"The Window"
//没有访问到局部变量,没有闭包的产生
// 类似于
//var f = object.getNameFunc();
//var f = function() {
    //return this.name;//此时的this指向的是window
//}
//f();
var name = "The Window";  
var object = {    
    name: "My Object",
    getNameFunc: function() {
        var that = this;
        return function() {
            return that.name;
        };
    }
};
console.log(object.getNameFunc()());//"My Object"
//访问到局部变量,有闭包的产生
// 类似于
//var f = object.getNameFunc();
//var f = function() {
    //return that.name;//此时的that指向的是object 
//}
//f();

6. 递归

6.1 什么是递归

递归: 如果一个函数在内部可以调用其本身,那么这个函数就是递归函数。简单理解:函数内部自己调用自己, 这个函数就是递归函数

注意: 递归函数的作用和循环效果一样,由于递归很容易发生“栈溢出”错误(stack overflow),所以必须要加退出条件return。

6.2 利用递归求1~n的阶乘

//利用递归函数求1~n的阶乘 1 * 2 * 3 * 4 * ..n
 function fn(n) {
     if (n == 1) { //结束条件
       return 1;
     }
     return n * fn(n - 1);
 }

6.3 利用递归求斐波那契数列

// 利用递归函数求斐波那契数列(兔子序列)  1、1、2、3、5、8、13、21...
// 用户输入一个数字 n 就可以求出 这个数字对应的兔子序列值
// 我们只需要知道用户输入的n 的前面两项(n-1 n-2)就可以计算出n 对应的序列值
function fb(n) {
  if (n === 1 || n === 2) {
        return 1;
  }
  return fb(n - 1) + fb(n - 2);
}

6.4 利用递归遍历数据

var data = [{
    id: 1,
    name: '家电',
    goods: [{
        id: 11,
        gname: '冰箱',
        goods: [{
            id: 111,
            gname: '海尔'
        }, {
            id: 112,
            gname: '美的'
        }, ]
    }, {
        id: 12,
        gname: '洗衣机'
    }]
}, {
    id: 2,
    name: '服饰'
}];
// 我们想要做输入id号,就可以返回的数据对象
// 1. 利用 forEach 去遍历里面的每一个对象
function getID(json, id) {
    var o = {};
    json.forEach(function(item) {
        // console.log(item); // 2个数组元素
        if (item.id == id) {
            o = item;
            // 2. 我们想要得里层的数据 11 12 可以利用递归函数
            // 里面应该有goods这个数组并且数组的长度不为 0 
        } else if (item.goods && item.goods.length > 0) {
            o = getID(item.goods, id);
        }
    });
    return o;
}

6.5 浅拷贝和深拷贝

  1. 浅拷贝只是拷贝一层,更深层次对象级别的只拷贝引用(地址)
// 浅拷贝只是拷贝一层, 更深层次对象级别的只拷贝引用.
var obj = {
    id: 1,
    name: 'andy',
    msg: {
        age: 18
    }
};
var o = {};

//for (var k in obj) {
//    // k 是属性名   obj[k] 属性值
//    o[k] = obj[k];
//}
//o.msg.age = 20;//o里面的msg拷贝的是obj里msg的地址,他们指向同一存储空间;所以当通过o改变msg时,obj里的msg也会发生改变
//console.log(obj.msg.age);//20

Object.assign(o, obj);
o.msg.age = 20;
console.log(obj.msg.age);//20
  1. 深拷贝拷贝多层,每一级别的数据都会拷贝
// 深拷贝拷贝多层, 每一级别的数据都会拷贝.
var obj = {
    id: 1,
    name: 'andy',
    msg: {
        age: 18
    },
    color: ['pink', 'red']
};
var o = {};
// 封装函数 
function deepCopy(newobj, oldobj) {
    for (var k in oldobj) {
        // 判断我们的属性值属于那种数据类型
        // 1. 获取属性值  oldobj[k]
        var item = oldobj[k];
        // 2. 判断这个值是否是数组(数组也是对象,所以要先判断是不是数组)
        if (item instanceof Array) {
            newobj[k] = [];
            deepCopy(newobj[k], item)
        } else if (item instanceof Object) {
            // 3. 判断这个值是否是对象
            newobj[k] = {};
            deepCopy(newobj[k], item)
        } else {
            // 4. 属于简单数据类型
            newobj[k] = item;
        }
    }
}
deepCopy(o, obj);
o.msg.age = 20;
console.log(obj.msg.age);//18
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值