ChatGPT解决这个技术问题 Extra ChatGPT

使用 jQuery 转义 HTML 字符串

有谁知道从 jQuery 中的字符串中转义 HTML 的简单方法?我需要能够传递任意字符串并将其正确转义以显示在 HTML 页面中(防止 JavaScript/HTML 注入攻击)。我确信可以扩展 jQuery 来做到这一点,但我目前对框架的了解还不够多,无法做到这一点。

另请参阅性能:jsperf.com/…

H
Hardik Modha

还有the solution from mustache.js

var entityMap = {
  '&': '&',
  '<': '&lt;',
  '>': '&gt;',
  '"': '&quot;',
  "'": '&#39;',
  '/': '&#x2F;',
  '`': '&#x60;',
  '=': '&#x3D;'
};

function escapeHtml (string) {
  return String(string).replace(/[&<>"'`=\/]/g, function (s) {
    return entityMap[s];
  });
}

很抱歉打扰,但无论如何这可以逆转吗?我不知道正则表达式,所以我需要帮助
A
Aliaksandr Sushkevich

由于您使用的是 jQuery,因此您只需设置元素的 text 属性:

// before:
// <div class="someClass">text</div>
var someHtmlString = "<script>alert('hi!');</script>";

// set a DIV's text:
$("div.someClass").text(someHtmlString);
// after: 
// <div class="someClass">&lt;script&gt;alert('hi!');&lt;/script&gt;</div>

// get the text in a string:
var escaped = $("<div>").text(someHtmlString).html();
// value: 
// &lt;script&gt;alert('hi!');&lt;/script&gt;

安全吗? linkedin.com/pulse/…
@paaacman 使用 jQuery 使用 .text().attr() 设置属性是安全的,但是像在该示例中那样构建 HTML 字符串肯定会遇到问题。
E
Edward
$('<div/>').text('This is fun & stuff').html(); // "This is fun &amp; stuff"

来源:http://debuggable.com/posts/encode-html-entities-with-jquery:480f4dd6-13cc-4ce9-8071-4710cbdd56cb


正如上面的答案中提到的,这个解决方案不能保证保留空白。
A
Alexander Elgin

如果您要转义 HTML,那么我认为只有三个是真正必要的:

html.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;");

根据您的使用案例,您可能还需要执行 "&quot; 之类的操作。如果列表足够大,我会使用一个数组:

var escaped = html;
var findReplace = [[/&/g, "&amp;"], [/</g, "&lt;"], [/>/g, "&gt;"], [/"/g, "&quot;"]]
for(var item in findReplace)
    escaped = escaped.replace(findReplace[item][0], findReplace[item][1]);

encodeURIComponent() 只会对 URL 进行转义,而不是 HTML。


如果有问题的 HTML 已经转义了实体,这个正则表达式会产生奇怪的结果。例如,转义“Tom & Jerry”将产生“Tom &amp; Jerry”
请使用 var 在本地声明 item;无论如何,在遍历数组时根本不要使用 for … in 循环!请改用普通的 for 循环。哦,它是 encodeURIComponent,而不是 escapeURIComponent
如果您正在使用标签属性,那么您还需要转义引号和/或双引号。 htmlspecialchars 的 PHP 文档包含一个有用的转换列表。 php.net/htmlspecialchars
只是对新人的善意提醒,如果您打算在您的网站上的某个地方使用非英语字符,请不要使用它......显然这不会因为带有重音符号的字符,例如 'é' : &eacute;以下是 html 实体列表,供参考:w3schools.com/tags/ref_entities.asp
@Ryan:虽然值得指出的是该解决方案不能正确处理已编码的字符串,但同样适用于本页上的大多数(可能是所有)解决方案也毫无价值。
c
chovy

很容易使用下划线:

_.escape(string) 

Underscore 是一个实用程序库,提供了许多原生 js 不提供的功能。还有 lodash,它与下划线的 API 相同,但被重写以提高性能。


逆是_.unescape(string)
z
zrajm

我写了一个很小的函数来做到这一点。它只转义 "&<>(但通常这就是您所需要的全部)。它比早先提出的解决方案稍微优雅一些,因为它只使用 one .replace() 来完成所有转换。 (编辑 2: 降低了代码复杂性,使函数变得更小更整洁,如果您对原始代码感到好奇,请参阅此答案的结尾。)

function escapeHtml(text) {
    'use strict';
    return text.replace(/[\"&<>]/g, function (a) {
        return { '"': '&quot;', '&': '&amp;', '<': '&lt;', '>': '&gt;' }[a];
    });
}

这是纯 Javascript,没有使用 jQuery。

转义 / 和 ' 太

编辑以回应 mklement 的评论。

上面的函数可以很容易地扩展为包含任何字符。要指定要转义的更多字符,只需将它们插入到正则表达式的字符类中(即在 /[...]/g 内)和作为 chr 对象中的条目。 (编辑 2: 也以同样的方式缩短了这个函数。)

function escapeHtml(text) {
    'use strict';
    return text.replace(/[\"&'\/<>]/g, function (a) {
        return {
            '"': '&quot;', '&': '&amp;', "'": '&#39;',
            '/': '&#47;',  '<': '&lt;',  '>': '&gt;'
        }[a];
    });
}

请注意上面使用 &#39; 表示撇号(符号实体 &apos; 可能已被使用——它是在 XML 中定义的,但最初并未包含在 HTML 规范中,因此可能并非所有浏览器都支持。请参阅: Wikipedia article on HTML character encodings)。我还记得在某处读到使用十进制实体比使用十六进制更广泛地支持,但我现在似乎无法找到它的来源。 (并且不可能有很多浏览器不支持十六进制实体。)

注意:/' 添加到转义字符列表中并不是很有用,因为它们在 HTML 中没有任何特殊含义,也不需要 被逃脱。

原始的 escapeHtml 函数

编辑 2: 原始函数使用变量 (chr) 来存储 .replace() 回调所需的对象。这个变量还需要一个额外的匿名函数来限定它,使函数(不必要地)更大更复杂。

var escapeHtml = (function () {
    'use strict';
    var chr = { '"': '&quot;', '&': '&amp;', '<': '&lt;', '>': '&gt;' };
    return function (text) {
        return text.replace(/[\"&<>]/g, function (a) { return chr[a]; });
    };
}());

我还没有测试过这两个版本中哪个更快。如果您这样做,请随时在此处添加有关它的信息和链接。


A
Adam Leggett

我意识到我参加这个聚会有多晚,但我有一个不需要 jQuery 的非常简单的解决方案。

escaped = new Option(unescaped).innerHTML;

编辑:这不会转义引号。唯一需要转义引号的情况是内容将被内联粘贴到 HTML 字符串中的属性。我很难想象这样做会是好的设计。

编辑3:对于最快的解决方案,请查看Saram的上述答案。这个是最短的。


i
intrepidis

这是一个干净、清晰的 JavaScript 函数。它将诸如“一些<许多”之类的文本转义为“一些<许多”。

function escapeHtmlEntities (str) {
  if (typeof jQuery !== 'undefined') {
    // Create an empty div to use as a container,
    // then put the raw text in and get the HTML
    // equivalent out.
    return jQuery('<div/>').text(str).html();
  }

  // No jQuery, so use string replace.
  return str
    .replace(/&/g, '&amp;')
    .replace(/>/g, '&gt;')
    .replace(/</g, '&lt;')
    .replace(/"/g, '&quot;')
    .replace(/'/g, '&apos;');
}

A
Aliaksandr Sushkevich

经过最后的测试,我可以推荐最快且完全跨浏览器兼容的原生 javaScript (DOM) 解决方案:

function HTMLescape(html){
    return document.createElement('div')
        .appendChild(document.createTextNode(html))
        .parentNode
        .innerHTML
}

如果您重复多次,您可以使用一次准备好的变量来执行此操作:

//prepare variables
var DOMtext = document.createTextNode("test");
var DOMnative = document.createElement("span");
DOMnative.appendChild(DOMtext);

//main work for each case
function HTMLescape(html){
  DOMtext.nodeValue = html;
  return DOMnative.innerHTML
}

看看我的最终表现comparison (stack question)。


有必要使用两个节点吗?一个怎么样:var p = document.createElement('p'); p.textContent = html; return p.innerHTML;
@DanDascalescu:根据MDNtextContent 功能仅受 Chrome 1+、Firefox 2、IE9、Opera 9.64 和 Safari 3 支持(后两者注释为“可能更早”)。因此,它将打破 OP“完全跨浏览器兼容”的主张。
p.innerText = html; return p.innerHTML
N
Nikita Koksharov

试试 Underscore.string lib,它适用于 jQuery。

_.str.escapeHTML('<div>Blah blah blah</div>')

输出:

'&lt;div&gt;Blah blah blah&lt;/div&gt;'

主下划线库现在具有 _.escape() 实用程序函数。
A
Alexander Elgin

escape()unescape() 旨在编码/解码 URL 的字符串,而不是 HTML。

实际上,我使用以下代码片段来完成不需要任何框架的技巧:

var escapedHtml = html.replace(/&/g, '&amp;')
                      .replace(/>/g, '&gt;')
                      .replace(/</g, '&lt;')
                      .replace(/"/g, '&quot;')
                      .replace(/'/g, '&apos;');

如果您要使用 ",那么您需要至少添加 ' 和 ``。这些只是 html 元素中的字符串标签数据真正需要的。对于 html 数据本身(外部标签),只需要前 3 个。
J
Jeena

我增强了 mustache.js 示例,将 escapeHTML() 方法添加到字符串对象。

var __entityMap = {
    "&": "&amp;",
    "<": "&lt;",
    ">": "&gt;",
    '"': '&quot;',
    "'": '&#39;',
    "/": '&#x2F;'
};

String.prototype.escapeHTML = function() {
    return String(this).replace(/[&<>"'\/]/g, function (s) {
        return __entityMap[s];
    });
}

这样就很容易使用"Some <text>, more Text&Text".escapeHTML()


有用,但我也将 __entityMap 移到函数本地范围内。并将所有这些包装到 if (typeof String.prototype.escapeHTML !== 'function'){...}
A
Alexander Elgin

如果您有 underscore.js,请使用 _.escape(比上面发布的 jQuery 方法更有效):

_.escape('Curly, Larry & Moe'); // returns: Curly, Larry &amp; Moe

W
Wayne

如果您要走正则表达式路线,则上面的 tghw 示例中存在错误。

<!-- WON'T WORK -  item[0] is an index, not an item -->

var escaped = html; 
var findReplace = [[/&/g, "&amp;"], [/</g, "&lt;"], [/>/g,"&gt;"], [/"/g,
"&quot;"]]

for(var item in findReplace) {
     escaped = escaped.replace(item[0], item[1]);   
}


<!-- WORKS - findReplace[item[]] correctly references contents -->

var escaped = html;
var findReplace = [[/&/g, "&amp;"], [/</g, "&lt;"], [/>/g, "&gt;"], [/"/g, "&quot;"]]

for(var item in findReplace) {
     escaped = escaped.replace(findReplace[item[0]], findReplace[item[1]]);
}

我认为应该是 for(var item in findReplace) { escaped = escaped.replace(findReplace[item][0], findReplace[item][1]); }
u
user427969

这是一个很好的安全示例...

function escapeHtml(str) {
    if (typeof(str) == "string"){
        try{
            var newStr = "";
            var nextCode = 0;
            for (var i = 0;i < str.length;i++){
                nextCode = str.charCodeAt(i);
                if (nextCode > 0 && nextCode < 128){
                    newStr += "&#"+nextCode+";";
                }
                else{
                    newStr += "?";
                }
             }
             return newStr;
        }
        catch(err){
        }
    }
    else{
        return str;
    }
}

您在那里抑制了哪些类型的异常?
r
raam86

您可以使用 vanilla js 轻松完成。

只需在文档中添加一个文本节点。它将被浏览器转义。

var escaped = document.createTextNode("<HTML TO/ESCAPE/>")
document.getElementById("[PARENT_NODE]").appendChild(escaped)

A
Alexander Elgin

2个不需要JQUERY的简单方法...

您可以像这样对字符串中的所有字符进行编码:

function encode(e){return e.replace(/[^]/g,function(e){return"&#"+e.charCodeAt(0)+";"})}

或者只是定位主要角色来担心 &、换行符、<>"',例如:

函数编码(r){ return r.replace(/[\x26\x0A\<>'"]/g,function(r){return"&#"+r.charCodeAt(0)+";"}) } var myString='编码 HTML 实体!\n"Safe" 转义