优先用 querySelector/querySelectorAll 替代老式 DOM 查找方法,因其语法统一、返回值一致且支持复杂选择器;innerHTML 用于插入含标签结构,textContent 仅更新纯文本更安全高效;批量操作 DOM 应用 DocumentFragment 或 innerHTML 一次性挂载,避免循环触发重排。
直接用 document.querySelector 和 document.querySelectorAll 替代老式 getElementById / getElementsByClassName,绝大多数场景下更简洁、可读性更强,且支持复杂选择器。
querySelector 比 getElementById 更值得默认使用不是因为它“高级”,而是它统一了查找逻辑:ID、类名、属性、伪类都能用同一套语法;而 getElementById 返回单个元素,getElementsByClassName 却返回 HTMLCollection(非数组),容易在调用 .forEach 时出错。
实操建议:
document.querySelector('#my-id'),和 getElementById 性能几乎无差别,但语义更一致document.querySelectorAll('.item'),返回的是 NodeList,可直接用 .forEach、.map
getElementsByClassName 获取后又转成数组——多此一举innerHTML 和 textContent 到底该选哪个关键看是否要解析 HTML 字符串。innerHTML 会触发 HTML 解析和 DOM 重建,有 XSS 风险且稍慢;textContent 只处理纯文本,安全、快、不触发重排。
常见错误现象:
innerHTML 渲染用户输入内容(如评论)→ 被执行恶意脚本innerHTML → 多余的解析开销,还可能意外清空绑定的事件监听器使用场景:
innerHTML
textContent
appendChild
每调用一次 appendChild

实操建议:
DocumentFragment 缓存所有新节点,最后一次性挂载:const frag = document.createDocumentFragment();
items.forEach(item => {
const el = document.createElement('div');
el.textContent = item;
frag.appendChild(el);
});
container.appendChild(frag);innerHTML 批量写入字符串(前提是内容可信),或用 insertAdjacentHTML 定位插入offsetHeight、getBoundingClientRect() 等触发重排的属性真正影响性能的往往不是“怎么选 API”,而是“在什么时机、以什么粒度去触碰 DOM”。批量、延迟、缓存 —— 这三个词比任何单个方法名都重要。