0

0

在JavaScript和React中安全渲染HTML字符串的教程

花韻仙語

花韻仙語

发布时间:2025-10-24 10:44:27

|

777人浏览过

|

来源于php中文网

原创

在JavaScript和React中安全渲染HTML字符串的教程

本教程旨在解决从数据库或变量中获取的html字符串被显示为纯文本的问题。我们将探讨在原生javascript中使用innerhtml以及在react框架中利用dangerouslysetinnerhtml属性来正确渲染html内容的方法。文章将详细解释这些机制的工作原理、提供示例代码,并重点强调使用这些功能时必须注意的跨站脚本(xss)安全风险及防范措施,确保在实现功能的同时保障应用安全。

理解HTML字符串渲染的挑战

在Web开发中,我们经常会遇到需要从后端服务、数据库或用户输入中获取包含HTML标签的字符串,并将其作为实际的HTML结构渲染到页面上。例如,一个存储在数据库中的问候语可能包含换行符

"hello, 
have a good day,"

然而,当直接将这样的字符串通过模板引擎或某些JavaScript方法插入到DOM中时,我们可能会发现
标签并没有被解释为换行,而是作为普通文本直接显示在屏幕上,例如:“hello,
have a good day,”。这是因为大多数现代前端框架和浏览器API默认会出于安全考虑对插入的内容进行HTML实体编码,以防止潜在的跨站脚本(XSS)攻击。

常见误区与原因分析

许多开发者在遇到这个问题时,可能会尝试以下几种方法:

  1. 直接在模板中绑定变量: 在许多模板语言或框架(如React的JSX、Vue的Mustache语法、Angular的插值表达式)中,直接使用{{ greeting }}这样的语法来显示变量内容时,默认行为通常是将变量值作为纯文本处理,并自动进行HTML转义。这意味着所有的HTML标签都会被转换为其对应的HTML实体(例如, 变为 youjiankuohaophpcn),从而避免了标签被浏览器解析。

    
    {{ greeting }}

    在这种情况下,即使greeting变量包含HTML标签,它们也会被安全地编码并显示为字面文本。

    立即学习Java免费学习笔记(深入)”;

  2. 尝试使用原生JavaScript的innerHTML(但方式不当): 开发者可能会想到使用JavaScript的innerHTML属性,因为它是原生DOM操作中用于设置元素HTML内容的标准方法。然而,如果使用方式不当,也可能无法达到预期效果。例如,以下代码尝试用一个固定的
    字符串替换元素内容:

    {{ greeting }}
    

    这段代码的问题在于,它将id="break"的元素内容替换为了一个固定的字符串
    ,而没有使用到我们想要渲染的greeting变量。此外,如果{{ greeting }}是框架的渲染机制,这段脚本可能在框架渲染之后执行,或者与框架的DOM管理冲突。

解决方案一:针对React应用的dangerouslySetInnerHTML

在React框架中,为了明确告知React我们需要插入未经转义的HTML字符串,并意识到这可能带来的安全风险,React提供了一个特殊的属性:dangerouslySetInnerHTML。

dangerouslySetInnerHTML 的使用

dangerouslySetInnerHTML 属性期望一个对象作为其值,该对象必须包含一个名为 __html 的键,其值就是你想要渲染的HTML字符串。

import React from 'react';

function MyComponent({ greeting }) {
  // 假设 greeting 是 "hello, 
have a good day," return ( ); } export default MyComponent;

解释:

星火作家大神
星火作家大神

星火作家大神是一款面向作家的AI写作工具

下载
  • dangerouslySetInnerHTML: 这个属性名称本身就包含了“危险”二字,旨在提醒开发者在使用时要格外小心。
  • {{ __html: greeting }}: React要求将HTML字符串包装在一个具有__html键的对象中。这是为了防止意外地将字符串直接赋值给dangerouslySetInnerHTML,强制开发者明确其意图。

通过这种方式,React会跳过其默认的HTML转义机制,直接将greeting变量中的HTML字符串作为DOM元素的内容插入到标签中,从而使
等标签被正确解析。

解决方案二:针对原生JavaScript的innerHTML

对于不使用React等前端框架,或者在纯原生JavaScript环境中操作DOM的场景,innerHTML属性是渲染HTML字符串的标准方法。

innerHTML 的使用

你可以通过获取DOM元素的引用,然后将其innerHTML属性设置为包含HTML的字符串。




原生JS渲染HTML








解释:

  • document.getElementById("greetingElement"):获取到需要操作的DOM元素。
  • element.innerHTML = greeting;:将greeting变量中的HTML字符串直接赋值给元素的innerHTML属性。浏览器会解析这个字符串并将其中的HTML标签渲染为实际的DOM结构。

安全警示:跨站脚本攻击 (XSS)

无论是使用React的dangerouslySetInnerHTML还是原生JavaScript的innerHTML,都存在严重的跨站脚本(XSS)安全风险。

什么是XSS? XSS攻击是指攻击者将恶意脚本注入到网页中,当其他用户访问该网页时,这些脚本就会在用户的浏览器上执行。如果你的HTML字符串来源于用户输入或不可信的第三方数据,攻击者可能会注入如下所示的恶意代码:

"Hello, "

如果直接渲染这段字符串,用户的浏览器就会执行alert('You are hacked!'),这只是一个简单的示例。更复杂的攻击可以窃取用户Cookie、会话令牌,甚至重定向用户到恶意网站。

防范措施:

  1. 输入消毒 (Sanitization): 在将任何来自用户或不可信源的HTML字符串渲染到页面之前,必须对其进行严格的消毒(Sanitization)。消毒是指移除或转义字符串中所有潜在的恶意标签和属性。

    • 使用成熟的库: 不要尝试自己编写HTML消毒函数,这非常复杂且容易出错。应使用经过广泛测试和验证的第三方库,例如:

      • DOMPurify (JavaScript): 一个高效且安全的HTML消毒库,可以在浏览器和Node.js环境中使用。
      • sanitize-html (Node.js): 另一个流行的服务器端HTML消毒库。
    • 示例 (使用DOMPurify):

      import DOMPurify from 'dompurify';
      
      const untrustedHTML = "@@##@@

      Hello

      "; const cleanHTML = DOMPurify.sanitize(untrustedHTML); // cleanHTML 现在是 "

      Hello

      ",恶意脚本已被移除 // 在React中
  2. 最小化使用: 尽量避免使用dangerouslySetInnerHTML或innerHTML。如果只需要简单的文本格式(如加粗、斜体、列表),可以考虑使用Markdown解析器将Markdown转换为HTML,或者通过CSS样式来控制文本呈现。

  3. 信任数据源: 只有当HTML字符串完全来自你信任的、且已经过严格验证和处理的后端系统时,才考虑直接渲染。

总结

渲染包含HTML标签的字符串是一个常见的需求,但必须谨慎处理。

  • 在React应用中,使用dangerouslySetInnerHTML={{ __html: yourHtmlString }}是实现此目的的官方方式,但其名称中的“dangerously”强调了潜在的安全风险。
  • 在原生JavaScript中,element.innerHTML = yourHtmlString;是标准方法。
  • 无论何种情况,最关键的是要对所有来自不可信源的HTML字符串进行严格的消毒(Sanitization),以防止跨站脚本(XSS)攻击。推荐使用如DOMPurify等成熟的库来完成消毒工作。

始终将安全性放在首位,确保你的应用程序在提供丰富内容的同时,也能够保护用户免受恶意攻击。

在JavaScript和React中安全渲染HTML字符串的教程

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

557

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

394

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

754

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

478

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

454

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1031

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

554

2023.09.20

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

1

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3万人学习

CSS教程
CSS教程

共754课时 | 21.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号