博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
js的一些基础(来自网络)
阅读量:4157 次
发布时间:2019-05-26

本文共 2346 字,大约阅读时间需要 7 分钟。

1、最简单的例子理解Javascript闭包

function greet(sth){    return function(name){        console.log(sth + ' ' + name);    }}//hi darrengreet('hi')('darren');

或者可以写成这样:

var sayHi = greet('hi');sayHi('darren');

 为什么greet的内部函数能使用sth这个变量?

其内部大致运作如下:
→ 创建全局上下文
→ 执行var sayHi = greet('hi');语句,创建greet上下文,变量sth存储在greet上下文中。
→ 继续执行greet函数内的语句,返回一个匿名函数,虽然greet上下文从堆栈上消失,但sth变量依旧存在于内存的某个空间。
→ 继续执行sayHi('darren');创建了sayHi上下文,并试图搜寻sth变量,但在sayHi这个上下文中没有sth变量。sayHi上下文会沿着一个作用域链找到sth变量对应的那个内存。 外层函数就像一个闭包,其内部函数可以使用外部函数的变量。

function buildFunctions(){    var funcArr = [];    for(var i = 0; i < 3; i++){        funcArr.push(function(){console.log(i)});    }    return funcArr;}var fs = buildFunctions();fs[0](); //3fs[1](); //3fs[2](); //3

以上,为什么结果不是0, 1, 2呢?
--因为i作为一个闭包变量,当前值为3,被内部函数使用。要实现想要的效果,可以在遍历的时候每一次遍历创建一个独立的上下文使其不受闭包影响。而自触发函数可以实现独立上下文。

function buildFunctions(){    var funcArr = [];    for(var i = 0; i < 3; i++){        funcArr.push((function(j){            return function(){              console.log(j);            };        }(i)));    }    return funcArr;}var fs = buildFunctions();fs[0](); //0fs[1](); //1fs[2](); //2

两个例子正好体现了闭包的2个方面:一个是内部函数使用闭包变量,另一个是把内部函数写在自触发函数中从而避免受闭包影响。

2、递归

  1. function factorial( n ){

  2.     return ( n <= 1 ) ? 1 : n * factorial( n-);
  3. }

3、call和apply的使用

在javascript OOP中,我们经常会这样定义:
      function cat(){
}
      cat.prototype={
food:"fish",
say: function(){
alert("I love "+this.food);}
}
var blackCat = new cat;
blackCat.say();

但是如果我们有一个对象whiteDog = {food:"bone"},我们不想对它重新定义say方法,那么我们可以通过call或apply用blackCat的say方法:blackCat.say.call(whiteDog);

所以,可以看出call和apply是为了动态改变this而出现的,当一个object没有某个方法,但是其他的有,我们可以借助call或apply用其它对象的方法来操作。

用的比较多的,通过document.getElementsByTagName选择的dom 节点是一种类似array的array。它不能应用Array下的push,pop等方法。我们可以通过:

var domNodes = Array.prototype.slice.call(document.getElementsByTagName("*"));
这样domNodes就可以应用Array下的所有方法了。

 

call 和 apply 都是为了改变某个函数运行时的 context 即上下文而存在的,换句话说,就是为了改变函数体内部 this 的指向。因为 JavaScript 的函数存在「定义时上下文」和「运行时上下文」以及「上下文是可以改变的」这样的概念。

二者的作用完全一样,只是接受参数的方式不太一样。例如,有一个函数 func1 定义如下:
var func1 = function(arg1, arg2) {};

就可以通过 func1.call(this, arg1, arg2); 或者 func1.apply(this, [arg1, arg2]); 来调用。其中 this 是你想指定的上下文,他可以任何一个 JavaScript 对象(JavaScript 中一切皆对象),call 需要把参数按顺序传递进去,而 apply 则是把参数放在数组里。

JavaScript 中,某个函数的参数数量是不固定的,因此要说适用条件的话,当你的参数是明确知道数量时,用 call,而不确定的时候,用 apply,然后把参数 push 进数组传递进去。当参数数量不确定时,函数内部也可以通过 arguments 这个数组来便利所有的参数。

转载地址:http://bwkxi.baihongyu.com/

你可能感兴趣的文章
Ubuntu系统root用户密码找回方法
查看>>
Linux驱动程序中比较重要的宏
查看>>
芯片驱动问题定位思路总结之一单板重启的问题
查看>>
S3C2440看门狗定时器
查看>>
LDD3源码分析之llseek分析
查看>>
linux read 用法
查看>>
LDD3源码分析之llseek分析(二)
查看>>
printk及控制台的日志级别
查看>>
Linux驱动加载实例
查看>>
Percona对MySQL标准版本的改进
查看>>
Instagram 架构师分析笔记
查看>>
你应该知道的Unix和Linux命令 lsof
查看>>
Nginx模块开发入门
查看>>
安全的Web主机iptables防火墙脚本
查看>>
Nginx的平滑重启和平滑升级
查看>>
大型互联网站解决海量数据的常见策略
查看>>
好多东西还是看官网的比较好... MySQL 5.1 Server System Variables
查看>>
chattr
查看>>
/etc/sysctl.conf 调优 & 优化Linux内核参数
查看>>
C代码阅读工具---calltrer
查看>>