优化 html 标签
2018年05月11日 08:56:24

阅读数:19

有些人写页面会走向一个极端,几乎页面所有的标签都用div,究其原因,用div有很多好处,一个是div没有默认样式,不会有margin、background等初始化设置,另外可能会觉得不用div还能用啥。所以看他的页面,一展开是div,再展开还是div,展开四、五层都是div。

这样对用户来说虽然没什么区别,但是作为一名有追求的程序员,这种写法看起来是比较难受的。有些人虽然知道html5新增了很多标签,但也不怎么去研究它的用法,反正用传统的div不是挺好的,为啥要去用兼容性不好的新东西。但是要是人人都这么想的话,新技术就不会发展了。有不一样或者更好的选择,当然要尝试下。

补充一点,并不是说使用div不好,该用div还是得用div,只是有其它更好的选择时应该选用其它的。

01选用合适的标签

1. 如果它是一段文字,那就用p标签吧,如果它是一个标题,那就用h1~h5标签吧,像下面这样:

<p class="title">你好,我是一个标题</p>

明明知道它是一个标题,为什么不用标题标签呢:

<h2>你好我是一个真实的标题</h2>

这样有个好处,让你的标签多样化,写样式的时候用标签选择器,就不用套很多的class

2. 如果它是一个表单,那就用form吧,用form用很多优点,我在《Effective前端1:能使用html/css解决的问题就不要使用JS》里面已经提到两点,第一点是自动表单提交,通过写一个form的action就能实现自动搜索跳转:

<form id="search-form" action="/search">
    <input type="search" name="keyword">
    <input type="number" name="price">
</form>

而不用自己去获取每个input的值,然后去拼参数跳转。

第二点是,自动监听回车事件提交,只要写一个form,里面有input,用户按回车就能自动提交,而不用自己去监听keypress事件处理。

除了这两点之外,写一个form还有一个很大的好处,当你用传统的jquery的选择器获取表单值的时候:

使用jquery获取表单数据XHTML

<div>
    <input id="user-name">
    <input id="password">
</div>
<script>
var userName = $("input#user-name").val(),
    password = $("input#password").val();
</script>

在这里,你为了获取两个表单数据,查了两次DOM,假设你有10个,就得查10个,如果是20个呢,对性能就会有影响了吧,特别是在移动端。但是如果你把div换成form,情况就不一样了:

<form id="register">
    <input name="user-name">
    <input name="password">
</form>

在获取表单数据时,可以这样用:

var form = document.forms.namedItem("register");
//var form = document.getElementById("register"); 或者这样写
var userName = form["user-name"].value,
    password = form.password.value;

只需要用原生的form属性就可以获取到表单的input,不用jquery,不用查DOM。同时这样有个弊端,就是当那个name的input不存在时,form.password是undefined,然后再获取value就挂了,但是这样早点暴露问题在某些情况下是不是更好呢

如果你用jQuery,还可以再增强一下,给表单对象添加一个函数:

添加一个jQuery对象的函数JavaScript

$.fn.serializeForm = function()
{
    var o = {};
    var a = this.serializeArray();
    $.each(a, function() {        if (o[this.name] !== undefined) {            if (!o[this.name].push) {
                o[this.name] = [o[this.name]];
            }
            o[this.name].push(this.value || '');
        } else {
            o[this.name] = this.value || '';
        }
    });    return o;
};

这个函数的作用在于借助jQuery的serializeArray获取所有的表单数据,然后再转化成一个object,以后需要获取这个表单的内容只需调一下:

var userData = $("#user-form").serializeForm();

即可,userData里面的属性就是input/select的name

3. html5 input

htm5提供了很多类型的input,使用这些input有很多好处,游览器会根据不同的input做出相应的优化,例如在iphone上,使用不同的input会弹出不同的键盘:

<input type=”email”>

<input type=”number”>

<input type=”tel”>

iphone弹不同类型的键盘,方便输入

即使在非html5浏览器也可以用,因为对不认识的type,浏览器会把它当作默认的text,唯一一个有兼容性的问题就是IE8、IE9会把不认识的input强制设置成text,即一访问页面,IE会把html里面的<input type="email">强制渲染成<input type="text">,这样就导致你没办法用CSS/JS根据这个type控制:

IE强制阉割不认识的input type为textCSS

/* will not work on ie8/ie9 */
input[type=email]{

}

不过,动态设置的type是可以生效的。初始化渲染的时候会被干掉。但是这个的影响应该不会很大。笔者还写了一个html5 input的表单验证插件,为了统一html5表单各浏览器处理不一致的问题,详见:实现跨浏览器html5表单验证

还有一个小坑,就是<input type="number">在Chrome下面是不允许输入逗号的,如果要支持输入逗号就不能用number了。

其它的:

(1) 如果内容是一个表格,那就用table,table有自适应的优点;

(2) 如果是加粗就用b/strong,而不是自己手动设置font-weight,这样有个好处,以后要更改字体,只需要设置b/strong的font-family,如果是手动写的font-weight,那就得一个个去找了;

(3) 如果是图片,那就用img,并且写上alt,帮助seo以及做为图片加载不出来显示的帮助文案;同时还可以用picture/srcset做响应式图片;

(4) 如果是输入框,就写个input,而不是自己写个p标签,然后设置contenteditable属性,因为这样在iphone上光标的定位会有问题;

(5) 如果是跳链那就写个a标签,而不是自己用javascript监听点击事件,然后自己做跳转。因为用a标签可以让搜索引擎爬取整个网站的内容,并且用a标签还有个好处,就是在手机端上滑的时候不会触发touchstart,如果自己做的跳转,要么用click,要么得搞个click增强库,笔者习惯这样写:

使用a标签做跳转XHTML

<ul>
    <li>
        <a style="display:block;color:inherit" href="/list?id=1">
            <img src="pic.jpg">
            <p>desc</p>
        </a>
    </li>
</ul>

02使用html5语义化标签

html5新增了很多语义化标签,一个传统的html4的页面结构:

html4页面结构XHTML

<ul class="nav">
    <li></li>
</ul>
<div class="header"></div>
<div class="main">
    <div class="section"></div>
    <div class="section"></div>
</div>
<div class="footer"></div>

可以用html5的新标签改装一下:

<nav></nav>
<header></header>
<main>
    <section></section>
    <section></section>
</main>
<footer></footer>

这样除了语义化的特点,更重要的是页面的组织发生了根本的变化,以前你在html4只能写一个h1标签,现在你可以写很多个h1标签。因为html5的页面大纲不再是根据h1、h2等标签进行划分,而是根据页面的章节。如下:

<body>
    <h1>Effective2</h1>
    <section>
        <h1>使用合适的标签</h1>
        <section><h2>使用form</h2></section>
        <section><h2>使用a标签</h2></section>
    </section>
    <section>
         <h1>使用h5标签</h1>
    </section>
</body>

搜索引擎会把这个页面概括为以下大纲:

Effective2

1. 使用合适的标签

1. 使用form

2. 使用a标签

2. 使用h5标签

读者可以从这个网站:html5 outliner进行实验(https://gsnedders.html5.org/outliner/)。

上面我们用了section进行划分章节,除了section之外,还有article、nav、aside,这四个标签可以互相嵌套划分层级关系,就像上面那样,section又嵌套了section,而每一个层级都可以任意使用h1~h5标签,同一个层级根据标题标签的主次进行划分。

这四个标签和div区别如下:

  1. div:作为一个普通的容器使用

  2. section:作为一个普通的章节

  3. article:适用于独立性较强的内容,例如笔者的这篇博客外层就用的article标签

  4. aside:可用作和页面主题内容相切的容器,像侧边栏、评论,像这篇博客底部的评论就用的aside

这篇博客无外乎说一件事件,站在浏览器的肩膀上进行开发,因为前端是根浏览器打交道的。这与第一篇《为何能用HTML/CSS解决的问题就不要使用JS?》的原理是相通的。

为什么说能使用html/css解决的问题就不要使用JS呢?两个字,因为简单。简单就意味着更快的开发速度,更小的维护成本,同时往往具有更好的体验,下面介绍几个实例。

1、导航高亮

导航高亮是一种很常见的问题,包括当前页面的导航在菜单里面高亮和hover时高亮。你可以用js控制,但是用一点CSS技巧就可以达到这个目的,不需要使用JS。

在正常态时,每个导航的默认样式为:

nav li{    opacity: 0.5;
}

当前页面的导航透明度为1. 为了实现这个目的:

首先通过body给不同的页面添加不同的类,用来标志不同的页面

<!-- home.html --><body class="home"></body>

<!-- buy.html --><body class="buy"></body>

所有的li也用class标志,为了有一个一一对应的关系:

<li class="home">home</li><li class="buy">buy</li>

然后就可以设置当前页面的样式,覆盖掉默认的样式:

body.home nav li.home,body.buy nav li.buy{    opacity: 1;
}

这样子,如果当前页面是home,则body.home na li.home这条规则生效,home的导航将高亮。

这个技艺在《精通CSS》这种书里面有提及。如果你用js控制,那么在脚本加载好之前,当前页面是不会高亮的,而脚本加载好之后突然就高亮了。所以用js吃力不讨好。

hover时的高亮,可以用css的:hover选择器:

nav li:hover{    opaciy: 1;
}

加上:hover选择器的优先级将会高于原本的,鼠标hover的时候将会覆盖默认样式,即高亮。

你也可以用mouse事件,mouseover的时候添加一个类,mouseleave的时候移除掉这个类,这样就变复杂了,用CSS甚至可以兼容不支持JS的浏览器,用户可能把浏览器的js禁掉了。我一个纯展示的静态页面,为啥要写js呢,是吧。

注意这个hover选择器特别好用,几乎适用于所有需要用鼠标悬浮时显示的场景。

2、鼠标悬浮时显示

鼠标悬浮的场景十分常见,例如导航的菜单:

以及在《Google地图开发总结》一文提到的,marker详情框的显示:

一般要把隐藏的东西如菜单作为hover目标的子元素或者相邻元素,才方便用css控制,例如上面的菜单,是把menu当作导航的一个相邻元素:

<li class="user">用户</li><li class="menu">
    <ul>
       <li>账户设置</li>
       <li>登出</li>
    </ul></li>

menu在正常态下是隐藏的:

.menu{    display: none;
}

而当导航hover时显示:

.user:hover + .menu{    display: list-item;
}

注意这里使用了一个相邻选择器,这也是上面说的为什么要写成相邻的元素。menu的位置可以用absolute定位。

同时menu自已本身hover的时候也要显示,否则鼠标一离开导航的时候,菜单就消失了:

.menu:hover{    display: list-item;
}

这里会有一个小问题,即menu和导航需要挨着一起,否则中间有空隙的话,上面添加的菜单hover就不能发挥作用了,但是实际情况下从美观的角度,两者是要有点距离的。这个其实也好解决,只要在menu上面再画一个透明的区域就好了,如下蓝色的方块:

可以用before/after伪类用absoute定位实现:

ul.menu:before{    content: "";    position: absolute;     left: 0;     top: -20px;     width: 100%;     height: 20px;      /**/}

如果我既写了css的hover,又监听了mouse事件,用mouse控制显示隐藏,双重效果会有什么情况发生,如果按正常套路,在mouse事件里面hover的时候,添加了一个display: block的style,会覆盖掉CSS的设置。

也就是说,只要hover一次,css的代码就不管用了,因为内联样式的优先级会高于外链的。

但是实际情况下会有意外发生,那就是在移动端iphone上面,触摸会触发CSS的hover,并且这个的触发会很高概率地先于touchstart事件,在这个事件里面会判断当前是显示还是隐藏的状态,由于css的hover发挥了作用,所以判断为显示,然后又把它隐藏了。

也就是说,点一次不出来,要点两次。所以最好别两个同时写。

第二种场景,使用子元素,这个更简单。把hover的目标和隐藏的对象当作同一个父容器的子元素,然后hover写在这个父容器上面就可以了,不用像上面那样,隐藏元素也要写个hover:

.marker-container .detail-info{    display: none
}.marker-container:hover .detail-info{    display: block
}

3、自定义radio/checkbox的样式

我们知道,使用原生的radio/checkbox是不可以改变它的样式的,得自己用div/span去画,然后再去监听点击事件。但是这样需要自己去写逻辑控制,例如radio只能选一个的功能,另一个是没有办法使用change事件。就是没有用原生的方便。

但是实际上可以用一点CSS3的技巧实现自定义的目的,如下,就是用原生实现的radio:

这个主要是借助了CSS3提供的一个伪类:checkd,只要radio/checkbox是选中状态,这个伪类就会生效,因此可以利用选中和非选中的这两种状态,去切换不同的样式。如下把一个checkbox和一个用来自定义样式的span写在一个label里面,checkbox始终隐藏:

<style>input[type=checkbox]{    display: none;
} /*未选中的checkbox的样式*/.checkbox{
}</style><label>
    <input type="checkbox">
    <span class="checkbox"></span></label>

写在label里面是为了能够点击span的时候改变checkbox的状态,然后再改一下选中态的样式即可:

input[type=checkbox]:checked + .checkbox{
}

关键在于这一步,添加一个打勾的背景图也好,使用图标字体也好。

:checked兼容性还是比较好的,只要你不用兼容IE8就可以使用,或者说只要你可以用nth-of-type,就可以用:checked。

4、多列等高

多列等高的问题是这样的,排成一行的几列由于内容长短不一致,导致容器的高度不一致:

你可以用js算一下,以最高的一列的高度去设置所有列的高度,然而这个会造成页面闪动,刚开始打开页面的时候高度不一致,然后发现突然又对齐了。这个解决办法主要有两种:

第一种是每列来一个很大的padding,再来一个很大的负的margin值矫正回去,就对齐了,如下:

<style>.wrapper > div{ float: left;        padding-bottom: 900px;        margin-bottom: -880px;        background-color: #ececec;        border: 1px solid #ccc;
    }</style><div class="wrapper"><div>column 1</div><div>column 2</div><div>column 3</div><div>column 4</div></div>

效果如下:

你会发现,这个对齐是对齐了,但是底部的border没有了,设置的圆角也不起作用了,究其原因,是因为设置了一个很大的padding值,导致它的高度变得很大,如上图所示。所以如果你想在底部absolute定位放一个链接”更多>>”也是实现不了了。

第二种办法是借助table的自适应特性 ,每个div都是一个td,td肯定是等高的,html结构不变,CSS改一下:

.wrapper{    display: table;    border-spacing: 20px;  /* td间的间距*/} .wrapper > div {     display: table-cell;     width: 1000px;          /*设置很大的宽度,table自动平分宽度 */border-radius: 5px;     /*这里设置圆角就正常了*/}

对齐效果如下:

这样还有一个好处,就是在响应式开发的时候,可以借助媒体查询动态地改变display的属性,从而改它排列的方式。例如在小于500px时,每一列占满一行,那么只要把display: table-cell覆盖掉就好了:

@media (max-width: 500px){    .wrapper{        display: block;
    }    .wrapper > div{        display: block;        width: 100%;
    }
}

效果如下所示:

如果在pad 1024px的设备上,希望一行显示2个,那应该怎么办呢?由于上面用的td,必定会排在同一行。其实可以在第二个和第三个中间加一个tr,让它换行:

<div class="wrapper">
 <div>column 1</div>
 <div>column 2</div>
 <span class="tr"></span>
 <div>column 3</div>
 <div>column 4</div></div>

在大屏和小屏时,tr是不显示的,而在中屏时,tr显示:

.tr{    display: none;
}

@media (max-width: 1024px) and (min-width: 501px){    .tr{        display: table-row;
    }
}

就能够实现在小屏时一行排两列了,只是这个有个小问题,就是在中屏拉到大屏的时候tr的dipslay: none已经没有什么作用,因为table的布局已经计算好。但是一般应该不用考虑这种拉伸范围很大的情况,正常刷新页面是可以的,如果真要解决那得借助下js。

5、 需要根据个数显示不同样式

例如说可能有1~3个item显示在同一行,而item的个数不一定,如果1个,那这个item占宽100%,2个时每一个50%,3个时每一个33%,这个你也可以用js计算一下,但是用CSS3就可以解决这个问题:

<style>
  li{
   width: 100%;
    }  
li:first-child:nth-last-child(2),    li:first-child:nth-last-child(2) ~ li{
width: 50%;
    }  
li:first-child:nth-last-child(3),    li:first-child:nth-last-child(3) ~ li{
 width: 33%;
    }  
</style><ul>
    <li>1</li>
    <li>2</li>
    <li>3</li></ul>

第5行的意思就是选择li的第一个元素,并且它是倒数第二个元素,第6行的意思是选择前面有是第一个且是倒数第二个li的所有li,第一行是选择了第一个,第二行选择除第一个外的其它所有元素。有三个元素的类似。

6、使用表单提交

提交请求有两种方式,一种是ajax,另外一种是表单提交。很多人都知道ajax,但往往忽略了还有个form提交。

假设在首页有一个搜索的表单,点击search的时候就跳到列表页

你可以一个个去获取所有的input的值,然后把它拼到网址参数重定向一下,但是其实可以不用这样,用一个表单提交就好了:

<form id="search-form" action="/search"><input type="search" name="keyword"><input type="number" name="price"></form>

将所有字段的名字写在input的name里面,然后form的action为搜索页的链接。这样子不用一行js代码就能够搜索跳转。

如果你需要做表单验证,那就监听submit事件,然后做验证,验证通过则调一下原生的submit就可以提交了,也是不需要手动去获取form的值。

7、自动监听回车事件

这个的场景是希望按回车的时候能够触发请求,像第6点,按回车实现跳转,或者是像下面的,按下回车就送一条聊天消息:

通常的做法是监听下keypress事件,然后检查一下keycode是不是回车,如果是则发请求。

但是其实有个特别简单的办法,也是不需要一行JS,那就是把表单写在一个form里面,按回车会自动触发submit事件。读者可以自己试试。这个就启示我们要用语义的html组织,而不是全部都用div。如果用相应的html标签,浏览器会自动做一些优化,特别是表单提交的input。

JS是万能的,几乎可以做任何事情,但是有时候会显得十分笨拙,在js/html/css三者间灵活地切换,往往会极大地简化开发,没有谁是最好的语言,只有适不适合。只要用得好,不管黑猫白猫,都是好猫。

优化 html 标签 为何能用HTML/CSS解决的问题就不要使用JS?的更多相关文章

  1. Effective前端1:能使用html/css解决的问题就不要使用JS

    div{display:table-cell;vertical-align:middle}#crayon-theme-info .content *{float:left}#crayon-theme- ...

  2. [转]能用HTML/CSS解决的问题就不要使用JS

    原文链接:http://www.codeceo.com/article/html-css-not-js.html 为什么说能使用html/css解决的问题就不要使用JS呢?两个字,因为简单.简单就意味 ...

  3. 能使用html/css解决的问题就不要使用JS

    为什么说能使用html/css解决的问题就不要使用JS呢?两个字,因为简单.简单就意味着更快的开发速度,更小的维护成本,同时往往具有更好的体验,下面介绍几个实例. 1. 导航高亮 导航高亮是一种很常见 ...

  4. Effective前端2:优化html标签

    div { float: left; } .keyboard > div + div { margin-left: 8px; } --> div{display:table-cell;ve ...

  5. 优化html标签

    借用Effective之名,开始写Effective系列,总结一些前端的心得. 有些人写页面会走向一个极端,几乎页面所有的标签都用div,究其原因,用div有很多好处,一个是div没有默认样式,不会有 ...

  6. img外头包着a时底部出现的一小段高度的解决方法。图片水平垂直居中用css解决的方法。

    <a><img/></a> 这种结构有时候在界面预览的时候会出现一段多出来的高度.这个高度,一开始我很奇怪是什么原因产生的.鼠标移动到a标签上会有高度出现,一开始我 ...

  7. DIV+CSS解决IE6,IE7,IE8,FF兼容问题(转至http://www.douban.com/note/163291324/)

    2011-07-25 21:11:47     DIV+CSS解决IE6,IE7,IE8,FF兼容问题 1.IE8下兼容问题,这个最好处理,转化成IE7兼容就可以.在头部加如下一段代码,然后只要在IE ...

  8. <转载>CSS解决图片过大撑破DIV的方法

    DIV+CSS网页内容中如果插入大于DIV层宽度显示,过大的图片将会撑破网页宽度显示从而网页严重变形,您是否遇到过?这里DIVCSS5给大家介绍几种解决图片撑破撑开网页DIV层方法. 图片撑破宽度解决 ...

  9. 用CSS解决一个让人头疼的问题

    需求:下面的文字内容分别都写在一个a标签里,现在需要获取到每一行最后一个a标签的竖线,并删除  我首先想到的是用CSS3新增选择器—— :nth-child()来解决,比如 :nth-child(3n ...

随机推荐

  1. HTML页面中解决内容元素随窗口变化布局变乱问题

    1.给body加上一个min-width最小宽度,以px为单位,这样当页面变小时,当达到你所设置的最小宽度,body的宽度不再改变,超出的部分会用横向滚动条显示,其内所有元素的布局也不会受影响. 2. ...

  2. SQL比较两表字段和字段类型

    一.问题 业务需要把TB_Delete_KYSubProject表数据恢复到TB_KYSubProject,但提示错误,错误原因是两表字段类型存在不一致 insert into [TB_KYSubPr ...

  3. 24. TABLES

    24. TABLES TABLES表提供有关数据库中表的信息. TABLES表有以下列: TABLE_CATALOG :表所属目录的名称.该值始终为def. TABLE_SCHEMA :表所属sche ...

  4. Oracle 11G RAC For ASM 利用RMAN COPY进行存储迁移

    转载请注明出处 一.需求背景 客户数据库存储空间接近存满,需购置一台新的存储,进行数据迁移,客户允许少量停机时间. 二.实施方法讨论 利用ASM rebalance 进行迁移 可以实现0宕机进行迁移, ...

  5. python基础003

    1. list 1.1 基础 list是一组有序的集合序列,可以包含任何类型且不必相同,并支持嵌套.采用如下创建方式: li = ["spam",2.0,5,[10,20]] 列表 ...

  6. Root of AVL Tree

    04-树5 Root of AVL Tree(25 分) An AVL tree is a self-balancing binary search tree. In an AVL tree, the ...

  7. [工具]Visual Studio

    1,Tab键的使用: 如不说有这样的代码:public Member member { get; set; } 当我们编辑完Member后,按一下Tab键,就能够将光标锁定到member上,等待键盘输 ...

  8. linux中的grep命令用法

    原文请移驾:http://blog.csdn.net/greytree/article/details/428532 grep -- print lines matching a pattern (将 ...

  9. 【02】使用Firebug查看和编辑HTML和CSS

    使用Firebug查看和编辑HTML和CSS 描述 在本章节的教程中,我们将讨论如何使用Firebug查看和编辑HTML和CSS. 使用Firebug查看和编辑HTML 在你要查看的元素上右击鼠标然后 ...

  10. 自动化项目配置或用例文件格式推荐--yaml

    关于yaml YAML语言的设计目标,就是方便人类读写.如果你想要实现一些用ini不好做到的配置,可以使用yaml格式作为配置文件 大小写敏感 使用缩进表示层级关系 缩进时不允许使用Tab键,只允许使 ...