IE8及更早版本无法识别<header>等HTML5语义标签,浏览器的DOM解析器会直接跳过这些未知元素,导致CSS无法匹配、JS查询返回null。解决方法是在<head>最顶部用document.createElement提前声明所有新标签,并配合display:block样式重置。
IE8 及更早版本压根不识别 <header>、<nav>、<section> 这类标签——它们在 DOM 里不存在,CSS 选不到,document.getElementById 或 querySelector 全部返回 null。不是样式没生效,是浏览器根本“看不见”。
document.createElement 必须在 <head> 最顶部执行旧 IE(尤其是 IE8)解析 HTML 时,遇到不认识的标签会直接跳过创建 DOM 节点。必须在它开始解析页面主体前,就让引擎“记住”这些标签名是合法的。否则,哪怕后面 JS 再调用 document.createElement('article'),已渲染的 <article> 元素也早已被忽略,无法补救。
<head> 内第一个 <script>,不能等 DOMContentLoaded 或 window.onload
['header','nav','article','section','aside','footer','main','figure','figcaption','time'].forEach(function(tag) { document.createElement(tag); });
main 很常见——IE11 以下也不认识它,但很多项目只注册了基础六种html5shiv 和手写脚本哪个更可靠html5shiv 不只是调用 document.createElement,它还注入了一组关键 CSS:article, aside, figcaption, figure, footer, header, hgroup, main, nav, section { display: block; }。只靠 JS 声明标签,IE8 仍默认给它们设 display: inline,导致布局全乱。
https://oss.maxcdn.com/html5shiv/3.7.2/html5shiv.min.js(比 cdnjs 更稳定)<!--[if lt IE 9]><script src="..."></script><![endif]-->,避免现代浏览器加载无用脚本if (!document.createElement('header').nodeName) { ['header','nav',...].forEach(...); } + 内联一段 <style>
即使标签能被识别、样式能生效,旧浏览器对现代 CSS 和 DOM API 的支持仍是断层。比如 querySelector 在 IE7 完全不可用,flexbox 在 IE10–IE11 用的是过时语法,localStorage 在 IE7 及以下要降级为 userData 行为。
header { ... },加一层 class 回退:.header, header { ... }
if ('querySelector' in document),否则用 getElementsByTagName 或 getElementsByClassName
html5shiv-printshiv.js,否则 @media print 里语义标签仍不显示解决旧浏览器兼容性问题的核心在于,从HTML解析到CSS匹配再到JS执行的全链路修复,重点关注标签声明、样式重置和功能检测的准确顺序。