第二章-在HTML中使用JavaScript

        事实上,WEB的核心语言是HTML,如何做到让JavaScript既可以与HTML页面共存,又不影响页面在其他浏览器中的呈现效果呢?

       要为WEB浏览器提供核心的脚本支持.

<script>元素

        在HTML页面中插入JavaScript的主要方法就是使用<script>元素,这个元素由Netscape创造并在Netscape Navigator 2中首先实现。后来,这个元素被加入到正式的HTML规范中。

        HTML 4.01为<script>定义了下列6个属性:

  • async:可选。异步,表示应该立即下载脚本,但不应该妨碍页面中的其他操作。比如下载文件资源和等待其他脚本。
  • charset:可选。表示通过src属性指定的代码的字符集。
  • defer:可选。表示脚本可以延迟到文档完全被解析和显示之后再执行,只对于外部脚本文件有效。
  • src:可选。表示包含要执行代码的外部文件
  • type:可选。可以看作为language(已被废弃的属性)的替代属性。表示编写代码使用的脚本语言的内容类型(也称为MIME类型)。虽然text/javascript和text/ecmascript都已经不被推荐使用,但人们一直以来使用的都还是text/javascript。实际上,服务器在传送JavaScript文件时使用的MIME类型通常是application/x-javascript。

使用<script>元素的方式有两种:

  1. 直接在页面中嵌入JavaScript代码
  2. 包含外部的JavaScript代码
直接在页面中嵌入JavaScript代码

        在HTML5之前,在页面中嵌入JavaScript代码的方法是

        <script type="text/javascript">
            function sayHi(){
                alert("Hi! ");
            }
        </script>

解释器会解释一个函数的定义,然后将该定义保存在自己的环境当中。在解释器对<script>元素内部的所有代码求值完毕以前,页面中的其余内容都不会被浏览器加载或显示。.

        <script type="text/javascript">
            function sayScript(){
                alert("</script>");
            }
        </script>

注意,在使用<script>嵌入JavaScript代码时,记住不要再代码中任何地方出现"<script>"字符串,比如上面的代码就会报错因为按照解析嵌入式代码的规则,当浏览器遇到字符串"</script>"时,就会认为那是结束的</script>标签。在前面通过添加转义字符“\”可以解决这个问题。

注意:在HTML5之后,不再需要指定type属性。

外部链接JavaScript文件
        <script src="example.js"></script>

         按照管理,外部JavaScript文件带有.js扩展名。但这个扩展名不是必需的,因为浏览器不会检查包含JavaScript的文件的扩展名。这样一来,使用JSP、PHP或其他服务器端语言动态生成JavaScript代码也就成为了可能。但是,服务器通常还是需要看扩展名决定为响应应用哪种MIME类型。如果不使用.js扩展名,请确保服务器能返回正确的MIME类型

        需要注意的是,带有src属性的<script>元素不应该在其<script>和</script>标签之间再包含额外的JavaScript代码。如果包含了嵌入的代码,则只会下载并执行外部脚本文件,嵌入的代码会被忽略。

        <script type="text/javascript" src="http://www.somewhere.com/afile.js"></script>

        另外,通过<script>元素的src属性还可以包含来自外部域的JavaScript文件。这一点既让<script>元素倍显强大,又让它备受争议。在这一点上,<script>与<img>元素非常相似,即它的src属性可以是指向当前HTML页面所在域之外的某个域中的完整URL。这样的话,位于外部域中的代码也会被加载和解析,就像这些代码位于加载它们的页面中一样。利用这一点就可以在必要时通过不同的域来提供JavaScript文件。不过,在访问自己不能控制的服务器上的JavaScript文件时则要多加小心。

        无论如何包含代码,只要不存在defer和async属性,浏览器都会按照<script>元素在页面中出现的先后顺序对它们进行以此解析。

标签的位置

按照传统的做法,所有<script>元素都应该放在页面的<head>元素中

        <! DOCTYPE html>
        <html>
          <head>
            <title>Example HTML Page</title>
            <script type="text/javascript" src="example1.js"></script>
            <script type="text/javascript" src="example2.js"></script>
          </head>
          <body>
            <! -- 这里放内容 -->
          </body>
        </html>

 这样做的目的就是把所有的外部文件(包括CSS文件和JavaScript文件)的引用都放在相同的地方

可是在文档的<head>元素中包含所有的js文件,意味着必须等到JavaScript代码都被下载、解析和执行完成后,才能开始呈现页面的内容,因为浏览器在遇到<body>标签时才开始呈现内容

所以,现代web application一般都把全部的js引用放在<body>元素中页面内容的后面

延迟脚本(defer)

HTML4为<script>标签定义了defer属性

这个属性的用途是表明脚本在执行时不会影响页面的构造

也就是说,脚本会被延迟到整个页面都解析完毕后再运行。

因此,在<script>元素中设置defer属性,相当于告诉浏览器立即下载,但延迟执行

        <! DOCTYPE html>
        <html>
          <head>
            <title>Example HTML Page</title>
            <script type="text/javascript" defer="defer" src="example1.js"></script>
            <script type="text/javascript" defer="defer" src="example2.js"></script>
          </head>
          <body>
            <! -- 这里放内容 -->
          </body>
        </html>

        注意:defer属性只适用于外部脚本文件

异步脚本(async)

        H5为<script>元素定义了async属性。这个属性与defer属性类似,都用于改变外部脚本文件的行为,并告诉浏览器立即下载文件

        但是与defer不同的是:标记为async的脚本并不会保证按照指定它们的先后顺序执行

        <! DOCTYPE html>
        <html>
          <head>
            <title>Example HTML Page</title>
            <script type="text/javascript" async src="example1.js"></script>
            <script type="text/javascript" async src="example2.js"></script>
          </head>
          <body>
            <! -- 这里放内容 -->
          </body>
        </html>

         以上代码中,第二个脚本文件可能会在第一个脚本文件之前执行

        因此,确保两者之间互不依赖非常重要。指定async属性的目的是不让页面等待两个脚本下载和执行,从而异步加载页面其他内容。为此,建议异步脚本不要在加载期间修改DOM。异步脚本一定会在页面的load事件前执行,但可能会在DOMContentLoaded事件触发之前或之后执行。

在XHTML中的用法

        可拓展超文本标记语言,即XHTML(Extensible HyperText Markup Language),是将HTML作为XML的应用而重新定义的一个标准

        编写XHTML代码的规则要比编写HTML严格得多,而且直接影响能否在嵌入JavaScript代码时使用<script>标签。以下面的代码块为例,虽然它们在HTML中是有效的,但在XHTML中则是无效的。

        <script type="text/javascript">
            function compare(a, b) {
                if (a < b) {
                    alert("A is less than B");
                } else if (a > b) {
                    alert("A is greater than B");
                } else {
                    alert("A is equal to B");
                }
            }
        </script>

        在HTML中,有特殊的规则用以确定<script>元素中的哪些内容可以被解析,但这些特殊的规则在XHTML中不适用。这里比较语句a < b中的小于号(<)在XHTML中将被当作开始一个新标签来解析。但是作为标签来讲,小于号后面不能跟空格,因此就会导致语法错误。

        避免在XHTML中出现类似语法错误的方法有两个

        1.用响应的HTML实体(&It)替换代码中的所有小于号

        2.用CData片段来包含JavaScript代码,在XML中,CData片段是文档的一个特殊区域,这个区域中可以包含不需要解析的任意格式的文本内容。因此,在CData片段中可以使用任意字符

        <script type="text/javascript"><! [CDATA[
            function compare(a, b) {
                if (a < b) {
                    alert("A is less than B");
                } else if (a > b) {
                    alert("A is greater than B");
                } else {
                    alert("A is equal to B");
                }
            }
        ]]></script>

 嵌入代码和外部文件

        使用外部文件的优点:

  • 可维护性:所有的JavaScript文件都放在一个文件夹里,维护起来就轻松多了
  • 可缓存:浏览器能够根据具体的设置缓存链接的所有外部JavaScript文件。也就是说,如果两个页面都是用同一个文件,那么这个文件只需下载一次。因此,最终结果就是能够加快页面开发速度
  • 适应未来:通过外部文件来包含JavaScript无须使用前面提到的XHTML或注释hack。

文档模式

        IE5.5引入文档模型的概念,而这个概念是通过使用文档类型(doctype)切换实现的

        最初的两种文档模式是:混杂模式(quirks mode)和标准模式(standards mode)

        混杂模式会让IE的行为与(包含非标准特性的)IE5相同,而标准模式则让IE的行为更接近标准行为

        在IE引入文档模式的概念后,其他浏览器也纷纷效仿。在此之后,IE又提出一种所谓的准标准模式(almost standards mode)。这种模式下的浏览器特性有很多都是符合标准的,但也不尽然。不标准的地方主要体现在处理图片间隙的时候(在表格中使用图片时问题最明显)。

        如果在文档开始处没有发现文档类型声明,则所有浏览器都会默认开启混杂模式

        对于标准模式,可以通过使用下面任何一种文档类型来开启

        <! -- HTML 4.01 严格型 -->
        <! DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN"
        "http://www.w3.org/TR/html4/strict.dtd">


        <! -- XHTML 1.0 严格型 -->
        <! DOCTYPE html PUBLIC
        "-//W3C//DTD XHTML 1.0 Strict//EN"
        "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">


        <! -- HTML 5-->
        <! DOCTYPE html>

        而对于准标准模式,则可以通过使用过渡型(transitional)或框架集型(frameset)文档类型来触发。

        准标准模式与标准模式非常接近

<noscript>元素

        早期浏览器都面临一个特殊的问题,即当浏览器不支持JavaScript时如何让页面平稳地退化。对这个问题的最终解决方案就是创造一个<noscript>元素,用以在不支持JavaScript的浏览器中显示替代的内容。这个元素可以包含能够出现在文档<body>中的任何HTML元素——<script>元素除外

        包含在<noscript>元素中的内容只有在下列情况下才会显示出来:

  • 浏览器不支持脚本
  • 浏览器支持脚本,但被禁用
        <html>
          <head>
            <title>Example HTML Page</title>
            <script type="text/javascript" defer="defer" src="example1.js"></script>
            <script type="text/javascript" defer="defer" src="example2.js"></script>
          </head>
          <body>
            <noscript>
              <p>本页面需要浏览器支持(启用)JavaScript。</p>
            </noscript>
          </body>
        </html>

         这个页面会在脚本无效的情况下向用户显示一条消息。而在启用了脚本的浏览器中,用户永远也不会看到它——尽管它是页面的一部分。

小结

        这一章,我们讨论了把JavaScript插入到HTML页面中要使用<script>元素。

        要注意的地方有:

  • 包含外部js文件时,必须将src属性设置为指向相应文件的URL
  • 所有的<script>元素都会按照它们在页面中出现的先后顺序一次被解析。在不使用defer和async属性的情况下,按照顺序解析
  • 由于浏览器会先解析完不使用defer属性的<script>元素中的代码,然后再解析后面的内容,所以一般应该把<script>元素放在页面最后,即主要内容后面,</body>标签前面
  • 使用defer属性可以让脚本在文档完全呈现之后再执行,延迟脚本总是按照指定他们的顺序执行
  • 使用异步属性可以表示当前脚本不必等待其他脚本,也不必阻塞文档呈现

OK,今天的读JavaScript高级程序设计就到这里了。