立即调用函数

目录一、了解立即调用函数表达式

二、立即调用函数表达式报错了?

三、使用立即调用函数的正确姿势

四、常见使用场景

写在最后

一、了解立即调用函数表达式

1.1 思维导图

1.2 什么是立即调用?

在详细了解这个之前,我们来谈了解一下“自执行”这个叫法,本文对这个功能的叫法也不一定完全对,每个人对他的理解都不一样,我们在这里用立即调用~

立即调用:顾名思义,该表达式一被创建就立即执行。

是一个在定义时就会立即执行的 JavaScript 函数。(function (x) {

console.log('x + x = ', x + x);})(5) // x + x = 10

这是一个被称为 自执行匿名函数 的设计模式,主要包含两部分:第一部分是包围在 圆括号运算符 () 里的一个匿名函数,这个匿名函数拥有独立的词法作用域。这不仅避免了外界访问此 IIFE 中的变量,而且又不会污染全局作用域。

第二部分再一次使用 () 创建了一个立即执行函数表达式,JavaScript 引擎到此将直接执行函数。

1.3 核心问题

当你声明一个函数的时候,通过在后面加括号就可以实现立即执行吗?var foo = function(){

console.log('余光');}(); // 余光 成功了!

// ...是不是意味着后面加个括弧都可以自动执行?function(){

console.log(''余光);}(); // Uncaught SyntaxError: Function statements require a function name// 什么?还需要一个函数名?不是叫 自执行匿名函数吗?// 我加上了函数名function foo(){

console.log('余光');}(); // Uncaught SyntaxError: Unexpected token ')'

很显然,例子中的第二条和第三条确实报错了,而且报错内容不一样,那么问题出现在哪呢?

二、立即调用函数表达式报错了?

有时,我们定义函数之后,立即调用该函数,这时不能在函数的定义后面直接加圆括号,这会产生语法错误。产生语法错误的原因是,function 这个关键字,既可以当做语句,也可以当做表达式,比如下边://语句function fn() {};//表达式var fn = function (){};

为了避免解析上的歧义,JS引擎规定,如果function出现在行首,一律解析成语句。因此JS引擎看到行首是function关键字以后,认为这一段都是函数定义,不应该以括号结尾,在它看来括号只是分组操作符。// 下面这个function在语法上是没问题的,但是依然只是一个语句// 加上括号()以后依然会报错,因为分组操作符需要包含表达式

function foo(){ /* code */ }(); // SyntaxError: Unexpected token )

// 但是如果你在括弧()里传入一个表达式,将不会有异常抛出// 但是foo函数依然不会执行function foo(){ /* code */ }( 1 );

// 因为它完全等价于下面这个代码,一个function声明后面,又声明了一个毫无关系的表达式: function foo(){ /* code */ }

( 1 );

三、使用立即调用函数的正确姿势

要解决上述问题,非常简单。

我们只需要用大括弧将代码的代码全部括住就行了,因为JavaScript里括弧()里面不能包含语句,所以在这一点上,解析器在解析function关键字的时候,会将相应的代码解析成function表达式,而不是function声明。

3.1 常见使用姿势// 下面2个括弧()都会立即执行(function () { /* code */ } ()); // 推荐使用这个(function () { /* code */ })(); // 但是这个也是可以用的

3.2 不常见的使用姿势(一)// 由于括弧()和JS的&&,异或,逗号等操作符是在函数表达式和函数声明上消除歧义的// 所以一旦解析器知道其中一个已经是表达式了,其它的也都默认为表达式了var i = function() {

console.log('余光')}(); // 余光true && function() {

console.log('余光')}(); // 余光0, function() { console.log('余光') }(); // 余光

3.3 不常见的使用姿势(二)// 如果你不在意返回值,或者不怕难以阅读// 你甚至可以在function前面加一元操作符号//转boolvar res1 = !function () {

console.log('余光');}()console.log('res1:', res1); // 余光 true// 转数字var res2 = +function () {

console.log('余光');}()console.log('res2:', res2); // 余光 NaN// 按位非var res3 = ~function () {

console.log('余光');}()console.log('res3:', res3); // 余光 NaN

3.4 不常见的使用姿势(三)

还有一个情况,使用new和void关键字,不过不太常见罢了。void function() {

console.log('余光');}();new function() {

console.log('余光');}();

四、常见使用场景

4.1 隔离作用域

IIFE最常见的功能,就是隔离作用域,在ES6之前JS原生也没有块级作用域的概念,所以需要函数作用域来模拟。

举例:var currentTime = (function () {

var time = new Date();

var year = time.getFullYear()

var month = time.getMonth()+1;

var date = time.getDate();

var hour = time.getHours();

var min = time.getMinutes();

return year + '-' + month + '-' + date + ' ' + hour + ':' + min;})()

你仍然可以在其他地方声明同名变量~

4.2 惰性函数

DOM事件添加中,为了兼容现代浏览器和IE浏览器,我们需要对浏览器环境进行一次判断:var addEvent = (function(){

if(window.addEventListener) {

return function(type, el, fn) {

el.addEventListener(type, fn, false);

}

}

else if(window.attachEvent) {

return function(type, el, fn) {

el.attachEvent('on' + type, fn);

}

}})();

4.3 用闭包保存状态

这里我仅举个例子,为我的下一篇文章——《JavaScript中的闭包》卖个关子var elems = document.getElementsByTagName('a');for (var i = 0; i < elems.length; i++) {

(function (lockedInIndex) {

elems[i].addEventListener('click', function (e) {

e.preventDefault();

alert('I am link #' + lockedInIndex);

}, 'false');

})(i);}

注意

当函数变成立即执行的函数表达式时,表达式中的变量不能从外部访问。(function () {

var name = "Barry";})();// 无法从外部访问变量 namename // 抛出错误:"Uncaught ReferenceError: name is not defined"

将 IIFE 分配给一个变量,不是存储 IIFE 本身,而是存储 IIFE 执行后返回的结果。var result = (function () {

var name = "Barry";

return name; })(); // IIFE 执行后返回的结果:result; // "Barry"相关免费学习推荐:javascript(视频)

js调用php函数兵每秒刷新,深入理解JavaScript立即调用函数表达式(IIFE)相关推荐

  1. js 对象深拷贝_这一次,彻底理解JavaScript深拷贝

    导语 这一次,通过本文彻底理解JavaScript深拷贝! 阅读本文前可以先思考三个问题: JS世界里,数据是如何存储的? 深拷贝和浅拷贝的区别是什么? 如何写出一个真正合格的深拷贝? 本文会一步步解 ...

  2. 彻底理解JavaScript中回调函数 (推荐)

    在javascript中回调函数非常重要,它们几乎无处不在.像其他更加传统的编程语言都有回调函数概念,但是非常奇怪的是,完完整整谈论回调函数的在线教程比较少,倒是有一堆关于call()和apply() ...

  3. 深入理解javascript系列(十七):函数柯里化

    之前的系列,我们介绍了什么是高阶函数.所有以函数作为参数的函数,都可以叫作高阶函数.并且我们常常利用高阶函数来封装一些公共逻辑. 本次,我们要继续学习,继续记录,柯里化.柯里化,其实就是高阶函数的一种 ...

  4. python rpc调用_从0到1:全面理解 RPC 远程调用

    上一篇关于 WSGI 的硬核长文,不知道有多少同学,能够从头看到尾的,不管你们有没有看得很过瘾,反正我是写得很爽,总有一种将一样知识吃透了的错觉. 今天我又给自己挖坑了,打算将 rpc 远程调用的知识 ...

  5. 理解javascript中的函数模式

    1.回调函数 将一个函数作为参数传给另一个函数,我们称它为回调函数 function test1(callback){//执行脚本//执行回调函数 callback();} 2.即时函数(自执行匿名函 ...

  6. matlab中ode45函数的用法_带你理解Excel中COUNTIF函数的简单用法

    每天5分钟,每天学一点. COUNTIF函数是Excel中最常用的统计函数之一,它的作用主要是用于根据特定条件对数据进行统计.假如,你想统计一下本周总共做了几次健身/瑜伽,或者统计上了几次培训课,那么 ...

  7. 【JavaScript】自定义函数

    函数 函数是一个可重用的代码块,用来完成某个特定功能.每当需要反复执行一段代码时,可以利用函数来避免重复书写相同代码. 函数包含着的代码只能在函数被调用时才会执行,就可以避免页面载入时执行该脚本 在J ...

  8. 一文探索 JavaScript 最强大特性—函数表达式

    函数是 ECMAScript 中最有意思的部分之一,这主要是因为函数实际上是对象.每个函数都是 Function 类型的实例,而 Function 也有属性和方法,跟其他引用类型一样.因为函数是对象, ...

  9. JS实现关闭当前子窗口,刷新父窗口及调用父窗口的方法

    一.JS实现关闭当前子窗口,刷新父窗口 JS代码如下: <script>  function refreshParent() {   window.opener.location.href ...

最新文章

  1. 微服务实战之春云与刀客(三)—— 面向接口调用代码结构实例
  2. 影像组学视频学习笔记(34)-使用3D Slicer软件提取影像组学特征、Li‘s have a solution and plan.
  3. pytorch优化器,学习率衰减学习笔记
  4. scala为什么需要函数
  5. Rightmost Digit
  6. Git安装及配置5分钟快速教程
  7. C++ 异常的详细介绍
  8. 设置zookeeper开机自启动
  9. 【Python】import自己的模块报错
  10. java层 native层_Java层的ServiceManager和Native层的ServiceManager的对应过程
  11. php如何打印程序运行时间,php计算程序运行时间的简单实例 - microtime
  12. 微信开发,自定义菜单不生效怎么办?重新关注也无效
  13. 边缘计算工作负载:虚拟机,容器还是裸机?
  14. 想转行做web前端工程师,必学这6大技能
  15. JDBC 连接 SQL2005 解决办法
  16. 怎样用计算机才能更快,如何让网速变快,详细教您怎么让电脑网速变快
  17. 【正则】详解python正则表达式之re.group()用法
  18. 安卓永久修改像素密度
  19. Scala zio-actors与akka-actor集成
  20. 网卡高可用与负载均衡【team】

热门文章

  1. sap权限激活_宅出职场含金量!SAP 解决方案培训课程线上免费学
  2. 低通滤波器转带通滤波器公式由来_?OPA627AP 高通滤波器
  3. python 代码分块_[代码全屏查看]-python多进程分块读取文件
  4. python爬取vip小说章节_python 爬取起点小说vip章节(失败)
  5. Linux 之一 基本命令
  6. linux可以打开浏览器嘛,Linux下怎样可以打开浏览器
  7. oracle tochar fm,oracle的to_char中的fm
  8. 小妲己智能机器人要连接wifi吗_“能扫能拖,聪明能干”,360扫地机器人S7一周体验分享...
  9. currenttimemillis 毫秒还是秒_Elasticsearch如何做到数十亿数据查询毫秒级响应?
  10. 百度安全 TrustZone SDK 正式成为 OP-TEE 官方推荐 Rust 开发环境