程序员人生 网站导航

前端编码风格规范之 HTML 规范

栏目:htmlcss时间:2015-01-20 08:49:19

HTML 规范


文档类型

推荐使用 HTML5 的文档类型申明: <!DOCTYPE html>.

(建议使用 text/html 格式的 HTML。避免使用 XHTML。XHTML 和它的属性,比如 application/xhtml+xml 在阅读器中的利用支持与优化空间都10分有限)。

HTML 中最好不要将无内容元素[1] 的标签闭合,例如:使用 <br> 而非<br />.


HTML 验证

1般情况下,建议使用能通过标准规范验证的 HTML 代码,除非在性能优化和控制文件大小上不能不做出妥协。

使用诸如 W3C HTML validator 这样的工具来进行检测。

规范化的 HTML 是显现技术要求与局限的显著质量基线,它增进了 HTML 被更好地应用。

不推荐

<title>Test</title>
<article>This is only a test.

推荐

<!DOCTYPE html>
<meta charset="utf⑻">
<title>Test</title>
<article>This is only a test.</article>

省略可选标签

HTML5 规范中规定了 HTML 标签是可以省略的。但从可读性来讲,在开发的源文件中最好不要这样做,由于省略标签可能会致使1些问题。

省略1些可选的标签确切使得页面大小减少,这很有用,特别是对1些大型网站来讲。为了到达这1目的,我们可以在开发后期对页面进行紧缩处理,在这个环节中这些可选的标签完全就能够省略掉了。


脚本加载

出于性能斟酌,脚本异步加载很关键。1段脚本放置在 <head> 内,比如<script src="main.js"></script>,其加载会1直阻塞 DOM 解析,直至它完全地加载和履行终了。这会造成页面显示的延迟。特别是1些重量级的脚本,对用户体验来讲那真是1个巨大的影响。

异步加载脚本可减缓这类性能影响。如果只需兼容 IE10+,可将 HTML5 的 async 属性加至脚本中,它可避免阻塞 DOM 的解析,乃至你可以将脚本援用写在 <head> 里也没有影响。

如需兼容老旧的阅读器,实践表明可以使用用来动态注入脚本的脚本加载器。你可以斟酌 yepnope 或 labjs。注入脚本的1个问题是:1直要等到 CSS 对象文档已就绪,它们才开始加载(短暂地在 CSS 加载终了以后),这就对需要及时触发的 JS 造成了1定的延迟,这多多少少也影响了用户体验吧。

终上所述,兼容老旧阅读器(IE9-)时,应当遵守以下最好实践。

脚本援用写在 body 结束标签之前,并带上 async 属性。这虽然在老旧阅读器中不会异步加载脚本,但它只阻塞了 body 结束标签之前的 DOM 解析,这就大大下降了其阻塞影响。而在现代阅读器中,脚本将在 DOM 解析器发现 body 尾部的 script 标签才进行加载,此时加载属于异步加载,不会阻塞 CSSOM(但其履行仍产生在 CSSOM 以后)。

所有阅读器中,推荐

<html>
  <head>
    <link rel="stylesheet" href="main.css">
  </head>
  <body>
    <!-- body goes here -->

    <script src="main.js" async></script>
  </body>
</html>

只在现代阅读器中,推荐

<html>
  <head>
    <link rel="stylesheet" href="main.css">
    <script src="main.js" async></script>
  </head>
  <body>
    <!-- body goes here -->
  </body>
</html>

语义化

根据元素(有时被毛病地称作“标签”)其被创造出来时的初始意义来使用它。打个比方,用 heading 元夙来定义头部标题,p 元夙来定义文字段落,用 a 元夙来定义链接锚点,等等。

有根据有目的地使用 HTML 元素,对可访问性、代码重用、代码效力来讲意义重大。

以下示例列出了1些的语义化 HTML 主要情况:

不推荐

<b>My page title</b>
<div class="top-navigation">
  <div class="nav-item"><a href="#home">Home</a></div>
  <div class="nav-item"><a href="#news">News</a></div>
  <div class="nav-item"><a href="#about">About</a></div>
</div>

<div class="news-page">
  <div class="page-section news">
    <div class="title">All news articles</div>
    <div class="news-article">
      <h2>Bad article</h2>
      <div class="intro">Introduction sub-title</div>
      <div class="content">This is a very bad example for HTML semantics</div>
      <div class="article-side-notes">I think I'm more on the side and should not receive the main credits</div>
      <div class="article-foot-notes">
        This article was created by David <div class="time">2014-01-01 00:00</div>
      </div>
    </div>

    <div class="section-footer">
      Related sections: Events, Public holidays
    </div>
  </div>
</div>

<div class="page-footer">
  Copyright 2014
</div>

推荐

<!-- The page header should go into a header element -->
<header>
  <!-- As this title belongs to the page structure it's a heading and h1 should be used -->
  <h1>My page title</h1>
</header>

<!-- All navigation should go into a nav element -->
<nav class="top-navigation">
  <!-- A listing of elements should always go to UL (OL for ordered listings) -->
  <ul>
    <li class="nav-item"><a href="#home">Home</a></li>
    <li class="nav-item"><a href="#news">News</a></li>
    <li class="nav-item"><a href="#about">About</a></li>
  </ul>
</nav>

<!-- The main part of the page should go into a main element (also use role="main" for accessibility) -->
<main class="news-page" role="main">
  <!-- A section of a page should go into a section element. Divide a page into sections with semantic elements. -->
  <section class="page-section news">
    <!-- A section header should go into a section element -->
    <header>
      <!-- As a page section belongs to the page structure heading elements should be used (in this case h2) -->
      <h2 class="title">All news articles</h2>
    </header>

    <!-- If a section / module can be seen as an article (news article, blog entry, products teaser, any other
     re-usable module / section that can occur multiple times on a page) a article element should be used -->
    <article class="news-article">
      <!-- An article can contain a header that contains the summary / introduction information of the article -->
      <header>
        <!-- As a article title does not belong to the overall page structure there should not be any heading tag! -->
        <div class="article-title">Good article</div>
        <!-- Small can optionally be used to reduce importance -->
        <small class="intro">Introduction sub-title</small>
      </header>

      <!-- For the main content in a section or article there is no semantic element -->
      <div class="content">
        <p>This is a good example for HTML semantics</p>
      </div>
      <!-- For content that is represented as side note or less important information in a given context use aside -->
      <aside class="article-side-notes">
        <p>I think I'm more on the side and should not receive the main credits</p>
      </aside>
      <!-- Articles can also contain footers. If you have footnotes for an article place them into a footer element -->
      <footer class="article-foot-notes">
        <!-- The time element can be used to annotate a timestamp. Use the datetime attribute to specify ISO time
         while the actual text in the time element can also be more human readable / relative -->
        <p>This article was created by David <time datetime="2014-01-01 00:00" class="time">1 month ago</time></p>
      </footer>
    </article>

    <!-- In a section, footnotes or similar information can also go into a footer element -->
    <footer class="section-footer">
      <p>Related sections: Events, Public holidays</p>
    </footer>
  </section>
</main>

<!-- Your page footer should go into a global footer element -->
<footer class="page-footer">
  Copyright 2014
</footer>

多媒体回溯

对页面上的媒体而言,像图片、视频、canvas 动画等,要确保其有可替换的接入接口。图片文件我们可采取成心义的备选文本(alt),视频和音频文件我们可以为其加上说明文字或字幕。

提供可替换内容对可用性来讲10分重要。试想,1位盲人用户如何能知晓1张图片是甚么,要是没有 @alt 的话。

(图片的 alt 属性是可不填写内容的,纯装潢性的图片便可用这么做:alt="")。

不推荐

<img src="luke-skywalker.jpg">

推荐

<img src="luke-skywalker.jpg" alt="Luke skywalker riding an alien horse">

尽可能用 alt 标签去描写图片,假想你需要对那些只能通过语音或看不见图片的用户表达图片究竟是甚么。

不推荐

<img src="huge-spaceship-approaching-earth.jpg" alt="Header image">

推荐

<img src="huge-spaceship-approaching-earth.jpg" alt="A huge spaceship that is approaching the earth">

关注点分离

理解 web 中如何和为什么辨别不同的关注点,这很重要。这里的关注点主要指的是:信息(HTML 结构)、外观(CSS)和行动(JavaScript)。为了使它们成为可保护的干净整洁的代码,我们要尽量的将它们分离开来。

严格地保证结构、表现、行动3者分离,并尽可能使3者之间没有太多的交互和联系。

就是说,尽可能在文档和模板中只包括结构性的 HTML;而将所有表现代码,移入样式表中;将所有动作行动,移入脚本当中。

在此以外,为使得它们之间的联系尽量的小,在文档和模板中也尽可能少地引入样式和脚本文件。

清晰的分层意味着:

  • 不使用超过1到两张样式表(i.e. main.css, vendor.css)
  • 不使用超过1到两个脚本(学会用合并脚本)
  • 不使用行内样式(<style>.no-good {}</style>
  • 不在元素上使用 style 属性(<hr style="border-top: 5px solid black">
  • 不使用行内脚本(<script>alert('no good')</script>
  • 不使用表象元素(i.e. <b>, <u>, <center>, <font>, <b>
  • 不使用表象 class 名(i.e. red, left, center)

不推荐

<!DOCTYPE html>
<html>
<head>
  <link rel="stylesheet" href="base.css">
  <link rel="stylesheet" href="grid.css">
  <link rel="stylesheet" href="type.css">
  <link rel="stylesheet" href="modules/teaser.css">
</head>
<body>
  <h1 style="font-size: 3rem"></h1>
  <b>I'm a subtitle and I'm bold!</b>
  <center>Dare you center me!</center>
  <script>
    alert('Just dont...');
  </script>
  <div class="red">I'm important!</div>
</body>
</html>

推荐

<!DOCTYPE html>
<html>
<head>
  <!-- Concatinate your style sheets into a single one -->
  <link rel="stylesheet" href="main.css">
</head>
<body>
  <!-- Don't use style attributes but assign sensible classes and apply styles in the stylesheet -->
  <h1 class="title"></h1>
  <!-- Don't use presentational elements and assign sensible classes -->
  <div class="sub-title">I'm a subtitle and I'm bold!</div>
  <!-- Maybe your comments get centered in your presentation but that decision is up to the stylesheet -->
  <span class="comment">Dare you center me!</span>
  <!-- You wanted to make it red because it's important so then also name the class important and decide in the stylesheet
   what you want to do with it -->
  <div class="important">I'm important!</div>

  <!-- Put all your scripts into files and concatinate them into a single one -->
  <script async src="main.js"></script>
</body>
</html>

HTML 内容至上

不要让非内容信息污染了你的 HTML。现在貌似有1种偏向:通过 HTML 来解决设计问题,这是明显是不对的。HTML 就应当只关注内容。

HTML 标签的目的,就是为了不断地展现内容信息。

  • 不要引入1些特定的 HTML 结构来解决1些视觉设计问题
  • 不要将 img 元素当作专门用来做视觉设计的元素

以下例子展现了误将 HTML 用来解决设计问题的这两种情况:

不推荐

<!-- We should not introduce an additional element just to solve a design problem  -->
<span class="text-box">
  <span class="square"></span>
  See the square next to me?
</span>
.text-box > .square {
  display: inline-block;
  width: 1rem;
  height: 1rem;
  background-color: red;
}

推荐

<!-- That's clean markup! -->
<span class="text-box">
  See the square next to me?
</span>
/* We use a :before pseudo element to solve the design problem of placing a colored square in front of the text content */
.text-box:before {
  content: "";
  display: inline-block;
  width: 1rem;
  height: 1rem;
  background-color: red;
}

图片和 SVG 图形能被引入到 HTML 中的唯1理由是它们显现出了与内容相干的1些信息。

不推荐

<!-- Content images should never be used for design elements!  -->
<span class="text-box">
  <img src="square.svg" alt="Square" />
  See the square next to me?
</span>

推荐

<!-- That's clean markup! -->
<span class="text-box">
  See the square next to me?
</span>
/* We use a :before pseudo element with a background image to solve the problem */
.text-box:before {
  content: "";
  display: inline-block;
  width: 1rem;
  height: 1rem;
  background: url(square.svg) no-repeat;
  background-size: 100%;
}

Type 属性

省略样式表与脚本上的 type 属性。鉴于 HTML5 中以上二者默许的 type 值就是 text/css 和 text/javascript,所以 type 属性1般是可以疏忽掉的。乃至在老旧版本的阅读器中这么做也是安全可靠的。

不推荐

<link rel="stylesheet" href="main.css" type="text/css">
<script src="main.js" type="text/javascript"></script>

推荐

<link rel="stylesheet" href="main.css">
<script src="main.js"></script>

可用性

如果 HTML5 语义化标签使用得当,许多可用性问题已引刃而解。ARIA 规则在1些语义化的元素上可为其添上默许的可用性角色属性,使用得当的话已使网站的可用性大部份成立。假设你使用 nav, aside, main,footer 等元素,ARIA 规则会在其上利用1些关联的默许值。 更多细节可参考 ARIA specification

另外1些角色属性则能够用来显现更多可用性情形(i.e.role="tab")。


Tab Index 在可用性上的应用

检查文档中的 tab 切换顺序并传值给元素上的 tabindex,这可以根据元素的重要性来重新排列其 tab 切换顺序。你可以设置 tabindex="⑴" 在任何元素上来禁用其 tab 切换。

当你在1个默许不可聚焦的元素上增加了功能,你应当总是为其加上tabindex 属性使其变成可聚焦状态,而且这也会激活其 CSS 的伪类:focus。选择适合的 tabindex 值,或是直接使用 tabindex="0" 将元素们组织成同1 tab 顺序水平,并强迫干预其自然浏览顺序。


微格式在 SEO 和可用性上的应用

如果 SEO 和可用性环境条件允许的话,建议斟酌采取微格式。微格式是通过在元素标签上申明1系列特定数据来达成特定语义的方法。

谷歌、微软和雅虎对如何使用这些额外的数据1定程度上的达成1致,如果正确的使用,这将给搜索引擎优化带来巨大的好处。

你可以访问 schema.org 取得更多内容细节。

看1个电影网站的简单例子:

不带微格式

<div>
 <h1>Avatar</h1>
 <span>Director: James Cameron (born August 16, 1954)</span>
 <span>Science fiction</span>
 <a href="../movies/avatar-theatrical-trailer.html">Trailer</a>
</div>

带有微格式

<div itemscope itemtype ="http://schema.org/Movie">
  <h1 itemprop="name">Avatar</h1>
  <div itemprop="director" itemscope itemtype="http://schema.org/Person">
  Director: <span itemprop="name">James Cameron</span> (born <span itemprop="birthDate">August 16, 1954)</span>
  </div>
  <span itemprop="genre">Science fiction</span>
  <a href="../movies/avatar-theatrical-trailer.html" itemprop="trailer">Trailer</a>
</div>

ID 和锚点

通常1个比较好的做法是将页面内所有的头部标题元素都加上 ID. 这样做,页面 URL 的 hash 中带上对应的 ID 名称,即构成描点,方便跳转至对应元素所处位置。

打个比方,当你在阅读器中输入 URL http://your-site.com/about#best-practices,阅读器将定位至以下 H3 上。

<h3 id="best-practices">Best practices</h3>

格式化规则

在每个块状元素,列表元素和表格元素后,加上1新空白行,并对其子孙元素进行缩进。内联元素写在1行内,块状元素还有列表和表格要另起1行。

(如果由于换行的空格引发了不可预计的问题,那将所有元素并入1行也是可以接受的,格式正告总好错误误正告)。

推荐

<blockquote>
  <p><em>Space</em>, the final frontier.</p>
</blockquote>

<ul>
  <li>Moe</li>
  <li>Larry</li>
  <li>Curly</li>
</ul>

<table>
  <thead>
    <tr>
      <th scope="col">Income</th>
      <th scope="col">Taxes</th>
    </tr>
  </thead>
  <tbody>
    <tr>
      <td>$ 5.00</td>
      <td>$ 4.50</td>
    </tr>
  </tbody>
</table>

HTML 引号

使用双引号(“”) 而不是单引号(“) 。

不推荐

<div class='news-article'></div>

推荐

<div class="news-article"></div>

[1]: 此处的空白元素指的是以下元素:area, base, br, col,command, embed, hr, img, input, keygen, link, meta, param,source, track, wbr


其他精彩文章

jQuery教程(19)-jquery ajax操作之序列化表单

jQuery教程(18)-ajax操作之履行POST要求

jQuery教程(20)-jquery ajax + php 操作之为Ajax要求提供不同...

jQuery教程(21)-jquery ajax 回调函数

jQuery教程(22)-ajax操作之毛病处理

jQuery教程(24)-ajax操作之Ajax和事件

更多关于android开发文章


------分隔线----------------------------
------分隔线----------------------------

最新技术推荐