首页 web前端 js教程 从function的定义看JavaScript的预加载

从function的定义看JavaScript的预加载

Nov 28, 2016 pm 03:26 PM

在JavaScript中定义一个函数,有两种写法:

1

2

function ftn(){} // 第一种

var ftn = function(){} // 第二种

登录后复制

有人说,这两种写法是完全等价的。但是在解析前,前一种写法会被解析器自动提升到代码的头部,因此违背了函数先定义后使用的原则,所以建议定义函数时候,全部采用后一种写法。

看完这句话,我第一个感觉是两个在使用时候是完全一致的,只是解析上有所差异。但是他的解释“前一种写法会被解析器自动提升到代码的头部”让我很困惑。

如是我有了下面第一个测试:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

<script type="text/javascript">

function ftn()

{

    alert(&#39;old&#39;);

}

  

var b = ftn;

  

function ftn()

{

    b();

    alert(&#39;new&#39;);

}

  

ftn();//浏览器报内存溢出

</script>

登录后复制

接下来我做了第二个测试:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

<script type="text/javascript">

var ftn = function()

{

    alert(&#39;old&#39;);

}

  

var b = ftn;

  

var ftn = function()

{

    b();

    alert(&#39;new&#39;);

}

  

ftn();//old,new依次弹出

</script>

登录后复制

网上的对这个解释是:第一种方式,刚开始其实没有重新定义ftn这个Function而在里面执行了其本身。第二种方式,ftn=function()这里没有执行到Function里面的代码ftn已经被重新定义了,所以这里的重定义是有效的。

如果不怎么清楚,那么我再做了一个下面的测试:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

<script type="text/javascript">

function ftn()

{

    alert(&#39;old&#39;);

}

  

var b = ftn;

  

function ftn()

{

    b();

    alert(&#39;new&#39;);

}

  

alert(b);//结果是重新定义的ftn内容

</script>

登录后复制

测试结果发现,重新定义ftn后,b的内容也会随着改变。下面我又做了两外一个测试:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

<script type="text/javascript">

var ftn = function()

{

    alert(&#39;old&#39;);

}

  

var b = ftn;

  

var ftn = function()

{

    b();

    alert(&#39;new&#39;);

}

  

alert(b);//结果是老的ftn内容

</script>

登录后复制

这样就很有意思了,在JavaScript里面除了基本数据类型,其他类型都是对象,对象是存在堆里面,它的别名是存在栈里面的地址,后一种测试很明显可以用这样的原理来理解。那么前面的测试为什么b会随着ftn的重新定义而改变了?

我有一种新解释,不知道对不对,在所有的讲JavaScript书里都会提到,JavaScript里面是没有方法重载的,后面定义的重名function会覆盖前面的function,var b = ftn;这句话是把b和ftn的引用指向同一个堆里面的内存,而重新定义function ftn(){}后,新的function对象覆盖了老的对象,而b和ftn引用的堆地址空间没变,如果真是这样,那么这种写法就合理了:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

<script type="text/javascript">

function ftn()

{

    alert(&#39;old&#39;);

}

  

var b = ftn;

  

var ftn = function()

{

    b();

    alert(&#39;new&#39;);

}

  

alert(b);//老的ftn

alert(ftn);//新的ftn

ftn();//old ,new

</script>

登录后复制

这样新的ftn在栈里面的地址改变了,指向了新的function对象的定义,而原来的function没有被覆盖,还保存,所以b还是老的ftn引用的地址。

刚刚写了一篇对JavaScript里面function理解的文章,回头再思考下我这边文章的内容,觉得自己通过测试的结果的理解还是有点问题,其实理解还是要从编译,运行的原理进行思考。JavaScript都是在执行代码时候才编译代码,因此我们var定义的类型可以不定,我们封装的对象还时候添加属性和方法,因此可以这么理解我标题所带来的问题,javascript一般的语言,例如定义一个变量var obj = new Object()只是做了一个很初的处理,在JavaScript里面叫做预编译,这种预编译的能力很弱,弱到你可以随便更改,而不会影响程序的运行,当对象被调用时候,JavaScript解释器才会进行编译,然后运行代码,这和java很不一样,java是先把代码编译好,调用他的时候才运行,这就是脚本语言的特点,所以脚本语言大多不快。但是当你这么定义函数:fonction ftn(){},这种就把代码进行了编译,也就是执行过了,这种写法和java函数的定义很相似了。这是我的新解,我觉得这个解释更加合理些。

JavaScript的“编译”只是检查有没有代码错误,不会运行代码,你可以试试在function里面随便写东西测试一下。预加载,先是function,再是var。上面的代码中,只有demo1和demo3受到影响,demo1和demo3的源代码顺序function - var - function,应用预加载后的顺序:function - function - var,预加载后的完整代码:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

<script type="text/javascript">

//demo1

function ftn()

{

    alert(&#39;old&#39;);

}

function ftn()

{

    b();

    alert(&#39;new&#39;);

}

var b = ftn;

  

ftn();//浏览器报内存溢出

</script>

登录后复制

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

<script type="text/javascript">

//demo3

function ftn()

{

  alert(&#39;old&#39;);

}

function ftn()

{

  b();

  alert(&#39;new&#39;);

}

var b = ftn;

  

alert(b);//结果是重新定义的ftn内容

</script>

登录后复制

预加载大概就这么样了。

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
2 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
2 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

在JavaScript中替换字符串字符 在JavaScript中替换字符串字符 Mar 11, 2025 am 12:07 AM

JavaScript字符串替换方法详解及常见问题解答 本文将探讨两种在JavaScript中替换字符串字符的方法:在JavaScript代码内部替换和在网页HTML内部替换。 在JavaScript代码内部替换字符串 最直接的方法是使用replace()方法: str = str.replace("find","replace"); 该方法仅替换第一个匹配项。要替换所有匹配项,需使用正则表达式并添加全局标志g: str = str.replace(/fi

8令人惊叹的jQuery页面布局插件 8令人惊叹的jQuery页面布局插件 Mar 06, 2025 am 12:48 AM

利用轻松的网页布局:8个基本插件 jQuery大大简化了网页布局。 本文重点介绍了简化该过程的八个功能强大的JQuery插件,对于手动网站创建特别有用

构建您自己的Ajax Web应用程序 构建您自己的Ajax Web应用程序 Mar 09, 2025 am 12:11 AM

因此,在这里,您准备好了解所有称为Ajax的东西。但是,到底是什么? AJAX一词是指用于创建动态,交互式Web内容的一系列宽松的技术。 Ajax一词,最初由Jesse J创造

10个JQuery Fun and Games插件 10个JQuery Fun and Games插件 Mar 08, 2025 am 12:42 AM

10款趣味横生的jQuery游戏插件,让您的网站更具吸引力,提升用户粘性!虽然Flash仍然是开发休闲网页游戏的最佳软件,但jQuery也能创造出令人惊喜的效果,虽然无法与纯动作Flash游戏媲美,但在某些情况下,您也能在浏览器中获得意想不到的乐趣。 jQuery井字棋游戏 游戏编程的“Hello world”,现在有了jQuery版本。 源码 jQuery疯狂填词游戏 这是一个填空游戏,由于不知道单词的上下文,可能会产生一些古怪的结果。 源码 jQuery扫雷游戏

如何创建和发布自己的JavaScript库? 如何创建和发布自己的JavaScript库? Mar 18, 2025 pm 03:12 PM

文章讨论了创建,发布和维护JavaScript库,专注于计划,开发,测试,文档和促销策略。

jQuery视差教程 - 动画标题背景 jQuery视差教程 - 动画标题背景 Mar 08, 2025 am 12:39 AM

本教程演示了如何使用jQuery创建迷人的视差背景效果。 我们将构建一个带有分层图像的标题横幅,从而创造出令人惊叹的视觉深度。 更新的插件可与JQuery 1.6.4及更高版本一起使用。 下载

使用AJAX动态加载盒内容 使用AJAX动态加载盒内容 Mar 06, 2025 am 01:07 AM

本教程演示了创建通过Ajax加载的动态页面框,从而可以即时刷新,而无需全页重新加载。 它利用jQuery和JavaScript。将其视为自定义的Facebook式内容框加载程序。 关键概念: Ajax和JQuery

如何为JavaScript编写无曲奇会话库 如何为JavaScript编写无曲奇会话库 Mar 06, 2025 am 01:18 AM

此JavaScript库利用窗口。名称属性可以管理会话数据,而无需依赖cookie。 它为浏览器中存储和检索会话变量提供了强大的解决方案。 库提供了三种核心方法:会话

See all articles