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

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

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

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

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

<script type="text/javascript">
function ftn()
{
alert('old');
} var b = ftn; function ftn()
{
b();
alert('new');
} ftn();//浏览器报内存溢出
</script>

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

<script type="text/javascript">
var ftn = function()
{
alert('old');
} var b = ftn; var ftn = function()
{
b();
alert('new');
} ftn();//old,new依次弹出
</script>

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

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

<script type="text/javascript">
function ftn()
{
alert('old');
} var b = ftn; function ftn()
{
b();
alert('new');
} alert(b);//结果是重新定义的ftn内容
</script>

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

<script type="text/javascript">
var ftn = function()
{
alert('old');
} var b = ftn; var ftn = function()
{
b();
alert('new');
} alert(b);//结果是老的ftn内容
</script>

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

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

<script type="text/javascript">
function ftn()
{
alert('old');
} var b = ftn; var ftn = function()
{
b();
alert('new');
} 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,预加载后的完整代码:

<script type="text/javascript">
//demo1
function ftn()
{
alert('old');
}
function ftn()
{
b();
alert('new');
}
var b = ftn; ftn();//浏览器报内存溢出
</script>
<script type="text/javascript">
//demo3
function ftn()
{
alert('old');
}
function ftn()
{
b();
alert('new');
}
var b = ftn; alert(b);//结果是重新定义的ftn内容
</script>

预加载大概就这么样了。

从function的定义看JavaScript的预加载的更多相关文章

  1. Javascript图片预加载详解

    预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发布 ...

  2. Javascript图片预加载详解 分类: JavaScript HTML+CSS 2015-05-29 11:01 768人阅读 评论(0) 收藏

    预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发布 ...

  3. 第一百五十七节,封装库--JavaScript,预加载图片

    封装库--JavaScript,预加载图片 首先了解一个Image对象,为图片对象 Image对象 var temp_img = new Image();   //创建一个临时区域的图片对象alert ...

  4. 再谈javascript图片预加载技术

    图片预加载技术的典型应用: 如lightbox方式展现照片,无疑需要提前获得大图的尺寸,这样才能居中定位,由于javascript无法获取img文件头数据,必须等待其加载完毕后才能获取真实的大小然后展 ...

  5. javascript 图片预加载

    <!DOCTYPE > <html xmlns="http://www.w3.org/1999/xhtml"> <head> <meta ...

  6. javascript图片预加载

    图片预加载是非常常见的一个功能,PC和移动端都会用到,尤其是在移动端,只要涉及到较多图片的加载都会用到该技术.下面是移动端用到的,引入了zepto. <!DOCTYPE html> < ...

  7. javaScript的预加载

    在有大量图片的页面中,为了避免页面加载完图片还未加载完成,我们通常会使用js的图片预加载. 这是一个预加载的demo: 首先把图片放入到一个类名为imgSrcArr的变量当中: var imgSrcA ...

  8. 利用CSS、JavaScript及Ajax实现图片预加载的三大方法

    预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发布 ...

  9. CSS和JavaScript以及Ajax实现预加载图片的方法及优缺点分析

    预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画 廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发 ...

随机推荐

  1. 基于Python的信用评分卡模型分析(一)

    信用风险计量体系包括主体评级模型和债项评级两部分.主体评级和债项评级均有一系列评级模型组成,其中主体评级模型可用“四张卡”来表示,分别是A卡.B卡.C卡和F卡:债项评级模型通常按照主体的融资用途,分为 ...

  2. impala 使用记录

    在命令行里面直接输入类似下面的语句,就可以执行impala sql语句. impala-shell -q "select * from xxxc limit 10;" 当用pyth ...

  3. dmesg命令详解

    基础命令学习目录 http://linux.cn/article-3587-1.html dmesg 命令的使用范例 下面我们展示一些最负盛名的‘dmesg’命令工具以及其实际使用举例.‘dmesg’ ...

  4. ifup,ifdown命令详解

    基础命令学习目录首页 原文链接:https://www.cnblogs.com/jing99/p/7881779.html ifup命令网络配置 ifup命令用于激活指定的网络接口.ifdown命令用 ...

  5. P4论文粗读笔记(一)

    一 文章名称:SNAP: Stateful Network-Wide Abstractions for Packet Processing 数据包处理的带状态网络概念 发表时间:2016 期刊来源:S ...

  6. 【搜索】POJ-2718 全排列+暴力

    一.题目 Description Given a number of distinct decimal digits, you can form one integer by choosing a n ...

  7. java属性编辑器,即PropertyEditor

    出处:http://www.iteye.com/topic/1123628

  8. [二叉树建树]1119. Pre- and Post-order Traversals (30) (前序和后序遍历建立二叉树)

    1119. Pre- and Post-order Traversals (30) Suppose that all the keys in a binary tree are distinct po ...

  9. Objective-C UIWebView添加脚视图

    - (void)addObserverForWebViewContentSize{ [self.webView.scrollView addObserver:self forKeyPath: cont ...

  10. sqlserver 修改表字段长度

    ALTER TABLE Table1 ALTER COLUMN column1 VARCHAR(255)