1.变量的作用域

前提:这里只全部都通过var创建的变量或对象

1.全局变量:函数外创建变量

  var x=10;
 function test(){
  alert("全局变量在test函数中"+x);
 function a(){
   alert("全局变量在私有函数中"+x); 
 }
  a();

}

2.局部变量:函数内部创建变量。

 

var x=10;
function test(){
  var y=20;
  alert("全局变量在test函数中"+x);//10
  alert("局部变量在test函数中"+y);//20
  alert("局部变量在test函数中"+z);//报错,z未定义
  function a(){
    var z=30;
    alert("局部变量早函数a中"+z);//30
  }
}
注:x书全部变量在整个作用域有效
  y是局部变量在test函数中有效,在a函数外无效
  z是局部变量只在a函数中有效
但是:如果我们想在函数外使用函数的变量怎么办?
这时候就产生了一个东西,闭包。

闭包的概念:是指有权访问另一个函数作用域中的变量的函数。

创建闭包的方式,就是在一个函数内部创建另一个内部(私有)函数。

下面是闭包的实例:

1.

function test(){
  var x=10;
  return function(){
    return x;
  }
}
alert(x);//调用局部变量x,报错未定义
//调用
var a=test();
  alert(a());
说明:a实际上就是闭包匿名函数,函数test中的局部变量x一直保存在内存中,并没有在test调用后被自动清除。

2.

var y;
function test(){
  var x=10;
  y=function(){
    return x;
  }
}
//调用函数
test();//undefined
//调用
alert(y());//10

3.

function test(arg){
  var y=function(){
    return arg;
  }
  arg++;
  return y;
}
var a=test(123);//undefined
alert(a());//124

4.

function test(){
  var x=[];
  var i;
  for(i=0;i<3;i++){
    x[i]=function(){
      return i;
    }
  }
  return x;
}
var a=test();//undefined
alert(a[0]());//3
alert(a[1]());//3
alert(a[2]());//3
说明:输出结果都是3,因为循环完毕i的值为3,三个函数都是指向这一个共同的值,
那么如何解决这个问题呢?下面我们来看。

解决方案一:使用自调函数来实现

function test(){
  var x=[];
  var i;
  for(i=0;i<3;i++){
    x[i]=(function(a){
      return function(){
        return a;
      }
    })(i);
  }
  return x;
}
var a=test();//undefined
alert(a[0]());//0
alert(a[1]());//1
alert(a[2]());//2

方案二:不使用自调函数来实现

function test(){
function makeClosure(x){
    return function(){
      return x;
    }
  }
  var a=[];
  var i;
  for(i=0;i<3;i++){
    a[i]=makeClosure(i);
  }
  return a;
}
var b=test();//undefined
alert(b[0]());//0
alert(b[1]());//1
alert(b[2]());//2

闭包应用实例--Getter与Setter

var getValue,setValue;
(function(){
  var secret=0;
  getValue=function(){
    return secret;
  };
  setValue=function(v){
    secret=v;
  }
})()
alert(getValue())//查看内部变量值
setValue(3)//设置内部局部变量的值
alert(getValue());

  迭代器实例

function setup(x){
  var i=0;
  return function(){
    return x[i++];
  }
}
var next=setup(["a","b","c"]);//undefined
alert(next());//a
alert(next());//b
alert(next());//c

javascript-原生-闭包的更多相关文章

  1. jQuery? 回归JavaScript原生API

    如今技术日新月异,各类框架库也是层次不穷.即便当年漫山红遍的JQuery(让开发者write less, do more,So Perfect!!)如今也有被替代的大势.但JS原生API写法依旧:并且 ...

  2. 深入理解JavaScript的闭包特性如何给循环中的对象添加事件

    初学者经常碰到的,即获取HTML元素集合,循环给元素添加事件.在事件响应函数中(event handler)获取对应的索引.但每次获取的都是最后一次循环的索引.原因是初学者并未理解JavaScript ...

  3. 据说每个大牛、小牛都应该有自己的库——JavaScript原生对象拓展

    在据说每个大牛.小牛都应该有自己的库——框架篇中我扬言要做个小牛,没想到一天没更新,小伙儿伴们就戏谑的问我,油哥是不是要太监了?其实事情是这个样子的,这不是太监的节奏,一是,关于写个自己的库的想法由来 ...

  4. JavaScript作用域闭包简述

    JavaScript作用域闭包简述 作用域 技术一般水平有限,有什么错的地方,望大家指正. 作用域就是变量起作用的范围.作用域包括全局作用域,函数作用域以块级作用域,ES6中的let和const可以形 ...

  5. JavaScript的闭包原理

    什么是js(JavaScript)的闭包原理,有什么作用? 一.定义 官方解释:闭包是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分. 个人的理解是 ...

  6. Js(javaScript)的闭包原理

    问题?什么是js(javaScript)的闭包原理,有什么作用? 一.定义 官方解释:闭包是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.  小编 ...

  7. 深入理解javascript的闭包

    闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域 ...

  8. JavaScript原生折叠扩展收缩菜单带缓冲动画

    JavaScript原生折叠扩展收缩菜单带缓冲动画 @落雨 <div id="div_two" style="display: none;"> &l ...

  9. 如何给循环中的对象添加事件--深入理解JavaScript的闭包特性

    初学者经常碰到的,即获取HTML元素集合,循环给元素添加事件.在事件响应函数中(event handler)获取对应的索引.但每次获取的都是最后一次循环的索引.原因是初学者并未理解JavaScript ...

  10. javascript,jquery(闭包概念)(转)

    偶尔听人说javascript闭包,让我联想起以前学编译原理和数字逻辑里讲的闭包,以前上课讲的闭包很难懂,而且含有递归的意思在里面,现在不想再查看里面的闭包概念. 但javascript我是经常要用, ...

随机推荐

  1. client-go实战之一:准备工作

    欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...

  2. docker数据卷(Data Volumes)

    Docker宿主机和容器之间文件拷贝docker copy 前言: Docker 数据管理 在生产环境中使用 Docker ,往往需要对数据进行持久化,或者需要在多个容器之间进行 数据共享,这必然涉及 ...

  3. docker入门及常用命令

    Docker简介 Docker 是一个开源的应用容器引擎,基于 Go 语言 并遵从Apache2.0协议开源. Docker 可以让开发者打包他们的应用以及依赖包到一个轻量级.可移植的容器中,然后发布 ...

  4. noip模拟42

    A. 卷 发现乘积足以爆 \(long\) \(long\),但是数据随机,可以略忽略精度问题 一个快速降低数的级别的方法是取对数,由于有性质 \(log(x * y)=logx+logy\),合并时 ...

  5. Vs code自动生成Doxygen格式注释

    前言 ​ 程序中注释的规范和统一性的重要性不言而喻,本文就推荐一种在用vscode编写代码时自动化生成标准化注释格式的方法,关于Doxygen规范及其使用可查看博文 代码注释规范之Doxygen. ​ ...

  6. Python - 面向对象编程 - 什么是对象和类

    面向对象编程 Object Oriented Programming,简称 OOP,是一种程序设计思想 OOP 把对象作为程序的基本单元,一个对象包含了数据和操作数据的方法 Python里面有一句话: ...

  7. Identity用户管理入门二(显示用户列表)

    在Controllers中新建AccountController,并在构造方法(函数)中注入SignInManager,UserManager UserManager   用户管理(注册,查找,修改, ...

  8. js判断是在移动端还是在pc端

    function chatQQ3(){ if(/Android|webOS|iPhone|iPod|BlackBerry/i.test(navigator.userAgent)) { //移动端打开 ...

  9. 技术栈:springboot2.x,vue,activiti5.22,mysql,带工作流系统

    前言 activiti工作流,企业erp.oa.hr.crm等审批系统轻松落地,请假审批demo从流程绘制到审批结束实例. 一.项目形式 springboot+vue+activiti集成了activ ...

  10. HTML+CSS+JS设计注册页面

    HTML实战--设计一个个人信息填写界面 应用的技术:HTML+CSS+JS CSS和JS是套用的模板,主要练习了表单的验证和正则表达式的使用 效果图: 代码: <!DOCTYPE html&g ...