js闭包用法实例详解


Posted in Javascript onDecember 13, 2016

本文实例讲述了js闭包用法。分享给大家供大家参考,具体如下:

引言

在公司中需要写一个js脚本来进行网站的统计,实现类似百度统计或者站长统计的功能,在实现的过程中自己感觉写的代码还是可以的,因为之前的js代码都是这些写,但是在组长代码走查的时候却非常的不满意,因为我们在js中写的方法都是全局的方法,因为我们写的东西需要嵌入到别人的界面中,所以这些全局的东西很可能会和别人的东西重名从而引发错误,所以说组长就给我留下一句话:用js闭包包起来。

变量作用域

我们都非常的熟悉变量的作用域就分为:全局变量和局部变量。js中在函数的内部可以直接读取全局变量。

Js代码

var n=999;
function f1(){
  alert(n);
}
f1(); // 999

另一方面,在函数外部自然无法读取函数内的局部变量。

Js代码

function f1(){
  var n=999;
}
alert(n); // error

我们还需要注意,在js中如果声明变量的时候一定要用var命令,否则实际上声明了一个全局的变量!

function f1(){
  n=999;
}
f1();
alert(n); // 999

如何从外部读取全局变量?

我们需要得到函数内部的全局变量该怎么办呢?。在正常情况下我们是做不到的,要想这么实现我们必须想一些办法——在函数的内部在定义一个函数:

function f1(){
  n=999;
  function f2(){
    alert(n); // 999
  }
}

在上面的代码中,函数f2就被包括在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1 就是不可见的。这就是JavaScript语言特有的“链式作用域”结构(chain scope),子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。

既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!

function f1(){
  n=999;
  function f2(){
    alert(n);
  }
  return f2;
}
var result=f1();
result(); // 999

闭包的概念

在上面的代码中f2函数,就是闭包。各种专业文献上的“闭包”(closure)定义非常抽象,很难看懂。我的理解是,闭包就是能够读取其他函数内部变量的函数。由于在Javascript语言中,只有函数内部的子函数才能读取局部变量,因此可以把闭包简单理解成“定义在一个函数内部的函数”。所以,在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。

下面来分享一下我们在项目中用到的js闭包的写法:

(function () {
  function getPageTitle() {
    return document.title;
  }
  function getBrowerLanguage() {
    var browerLanguage = !navigator.browserLanguage ? navigator.language : navigator.browserLanguage;
    return browerLanguage;
  }
  /**
   * 当前页面地址#后的部分
   */
  function getLastUrl() {
    var url = window.location.hash;
    if (!url) {
      return null;
    }
    else {
      return url.toString().split("#")[1];
    }
  }
  function GetRandomNum() {
    var arr = document.cookie.match(new RegExp("(^| )" + "statisticssCookie=([^;]*)(;|$)"));
    if (arr != null) {
      return arr[2];
    } else {
      var tempRandomNum = guid();
      document.cookie = "statisticssCookie = " + tempRandomNum;
      return tempRandomNum;
    }
  }
  function guid() {
    function S4() {
      return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
    }
    return (S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4());
  }
  function addJs() {
    var url = "http://localhost:10086/tongji/tongji/do?title=" + getPageTitle() + "&browerLanguage=" + getBrowerLanguage() + "&lastUrl=" + getLastUrl() + "&upFlag=" + GetRandomNum();
    var head = document.getElementsByTagName('head')[0];
    var js = document.createElement('script');
    js.type = 'text/javascript';
    js.src = url;
    head.appendChild(js);
  }
  window.statistics = addJs;//将addJs这个方法挂在window下面,这样在外界是可以访问的,否则外界永不能访问到我写的方法
  document.ready = addJs();//DOM树加载完成后执行
})(window)

小结

在开始的时候组长就问我会不会js闭包函数,因为如果不会闭包写不出很好的代码。写的代码都非常的粗糙,所以说我们在写代码的时候,不能仅仅满足功能实现,而且需要考虑一些其他方面的东西。当然js闭包在使用的时候也有一些弊端,所以我们在使用的时候也需要综合全面的信息考虑。

希望本文所述对大家JavaScript程序设计有所帮助。

Javascript 相关文章推荐
JQuery index()方法使用代码
Jun 02 Javascript
网易JS面试题与Javascript词法作用域说明
Nov 09 Javascript
JS+CSS实现经典的左侧竖向滑动菜单效果
Sep 23 Javascript
使用JQuery实现Ctrl+Enter提交表单的方法
Oct 22 Javascript
JS自定义混合Mixin函数示例
Nov 26 Javascript
bootstrap table之通用方法( 时间控件,导出,动态下拉框, 表单验证 ,选中与获取信息)代码分享
Jan 24 Javascript
从零学习node.js之mysql数据库的操作(五)
Feb 24 Javascript
Angularjs验证用户输入的字符串是否为日期时间
Jun 01 Javascript
浅谈angular表单提交中ng-submit的默认使用方法
Sep 30 Javascript
利用Node.js如何实现文件循环覆写
Apr 05 Javascript
小程序自动化测试的示例代码
Aug 11 Javascript
详解JavaScript中Arguments对象用途
Aug 30 Javascript
深入学习Bootstrap表单
Dec 13 #Javascript
ThinkJS中如何使用MongoDB的CURD操作
Dec 13 #Javascript
Bootstrap Img 图片样式(推荐)
Dec 13 #Javascript
Javascript oop设计模式 面向对象编程简单实例介绍
Dec 13 #Javascript
教大家轻松制作Bootstrap漂亮表格(table)
Dec 13 #Javascript
AngularJS自定义控件实例详解
Dec 13 #Javascript
Node.js中process模块常用的属性和方法
Dec 13 #Javascript
You might like
VIM中设置php自动缩进为4个空格的方法详解
2013/06/14 PHP
PHP单例模式详细介绍
2015/07/01 PHP
php mongodb操作类 带几个简单的例子
2016/08/25 PHP
PHP中仿制 ecshop验证码实例
2017/01/06 PHP
详解配置 Apache 服务器支持 PHP 文件的解析
2017/02/15 PHP
PHP对称加密算法(DES/AES)类的实现代码
2017/11/14 PHP
Jquery在IE7下无法使用 $.ajax解决方法
2009/11/11 Javascript
基于jquery的合并table相同单元格的插件(精简版)
2011/04/05 Javascript
JS中Iframe之间传值及子页面与父页面应用
2013/03/11 Javascript
javascript的数组和常用函数详解
2014/05/09 Javascript
jQuery控制的不同方向的滑动(向左、向右滑动等)
2014/07/18 Javascript
Javascript基础知识盲点总结之函数
2016/05/15 Javascript
js接收并转化Java中的数组对象的方法
2016/08/11 Javascript
Vue.js双向绑定实现原理详解
2016/12/22 Javascript
微信小程序 页面跳转传递值几种方法详解
2017/01/12 Javascript
基于canvas的二维码邀请函生成插件
2017/02/14 Javascript
Vue中props的使用详解
2018/06/15 Javascript
详解vue使用vue-layer-mobile组件实现toast,loading效果
2018/08/31 Javascript
vue配置font-awesome5的方法步骤
2019/01/27 Javascript
微信小程序实现日期格式化和倒计时
2020/11/01 Javascript
解决layui checkbox 提交多个值的问题
2019/09/02 Javascript
使用Vue.set()方法实现响应式修改数组数据步骤
2019/11/09 Javascript
AI小程序之语音听写来了,十分钟掌握百度大脑语音听写全攻略
2020/03/13 Javascript
前端 javascript 实现文件下载的示例
2020/11/24 Javascript
使用Python的Scrapy框架编写web爬虫的简单示例
2015/04/17 Python
python MySQLdb Windows下安装教程及问题解决方法
2015/05/09 Python
Python网络编程使用select实现socket全双工异步通信功能示例
2018/04/09 Python
Python2和Python3的共存和切换使用
2019/04/12 Python
利用Python小工具实现3秒钟将视频转换为音频
2019/10/29 Python
英国汽车座椅和婴儿车购物网站:Uber Kids
2017/04/19 全球购物
销售人员自我评价
2014/02/01 职场文书
亲戚结婚的请假条
2014/02/11 职场文书
2014厂务公开实施方案
2014/02/17 职场文书
国庆65周年演讲稿:回首往昔,展望未来
2014/09/21 职场文书
Python爬虫进阶之Beautiful Soup库详解
2021/04/29 Python
CSS使用伪类控制边框长度的方法
2022/01/18 HTML / CSS