登录
首页 >  文章 >  前端

HTML模板标签实现列表渲染方法

时间:2026-04-23 13:45:55 443浏览 收藏

HTML 的 `

HTML中template列表渲染 HTML中template标签结合原生JS

template 标签本身不渲染、不执行、不绑定,它只是个“离线仓库”——想用它做列表渲染,必须靠 JS 手动克隆、填值、插入。直接写 for 或插值语法(如 {{name}})完全无效,浏览器会原样忽略。

为什么 template.content.cloneNode(true) 是唯一可靠入口

很多人卡在第一步:用 document.querySelector('template').innerHTML 拿不到内容,返回空字符串。这是因为 template 的真实内容被包裹在 DocumentFragment 里,只能通过 content 属性访问。

  • template.innerHTML 始终为空,永远别用
  • template.textContent 返回所有文本(含换行缩进),但无法操作结构
  • 必须走 template.content.cloneNode(true),否则后续所有节点操作都失效
  • 不加 true 参数是浅克隆,子元素事件监听器、表单状态、input 光标位置等全丢

填值时别碰 innerHTML,优先用 textContentvalue

innerHTML 拼接字符串填值,等于主动引入 XSS 风险,还会意外销毁已绑定的事件监听器(比如模板里原本有 button,克隆后 onclick 就没了)。

  • 纯文本内容:用 el.querySelector('.title').textContent = data.title
  • 表单控件:el.querySelector('input').value = data.value,不是 textContent
  • 需要保留 HTML 结构?先对用户输入做转义(如 text.replace(/&/g, '&')),再塞进 innerHTML
  • 模板中含 cloneNode(true) 不会执行脚本,也不会重复加载样式,别指望它“活”起来

批量插入必须用 DocumentFragment,否则性能崩得快

循环里每轮都 container.appendChild(cloned),等于让浏览器反复重排重绘。100 条数据可能就明显卡顿;500 条基本卡死。

  • const frag = document.createDocumentFragment()
  • 循环体内:克隆 → 填值 → frag.appendChild(cloned)
  • 循环结束后,只调一次 container.appendChild(frag)
  • frag 是轻量中间容器,不能被 querySelector 查到,也不在 DOM 树中,放心用

data-属性能留,slot 不生效,别混用 Web Components 逻辑

想存 ID 或索引?data-id 可以保留在克隆节点上,但不会自动变成 JS 对象属性;想用 slot 分发内容?原生 template 不激活 slot 机制——它没挂载到 Shadow DOM,slot="xxx" 根本不解析。

  • 存元信息:用 cloned.querySelector('li').dataset.id = item.id
  • 避免 cloned.querySelector('[slot="title"]') 这类写法,永远查不到目标
  • 如果真需要 slot 行为,得配合 customElements.define() + attachShadow(),那已是 Web Components 范畴,和纯 template 渲染无关

最容易被忽略的是:每次克隆都得是全新副本。共用一个节点引用、漏掉 cloneNode(true)、或在循环外提前取了 content,都会导致只显示最后一项,或者 DOM 状态错乱。这不是 bug,是 template 的设计使然——它不维护响应式,只提供干净的原始结构。

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于文章的相关知识,也可关注golang学习网公众号。

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>