草庐IT

php - 如何将带有HTML实体和无效字符的文本转换为其等效的UTF-8?

coder 2024-04-14 原文

我之所以更改标题,是因为我不知道是什么特殊的windows字符导致了我的问题,使问题看起来像是重复的。
如何将HTML实体、类型为&35;[0-9]+;和&x[a-fa-f0-9]+;的字符引用、无效字符引用和无效的Windows字符CHR(151)转换为其UTF-8等效字符?
基本上如何清理一些非常糟糕的变量编码文本并将其保存为utf-8?
原题如下
转换&[0-9]+;和&x[a-fa-f0-9]+;对utf-8等价物的引用?
例如

—
—


-
就像浏览器一样,但是有了php。
编辑:即使是windows制作但浏览器仍然显示的非标准版本。

最佳答案

用我最后使用的解决方案回答我自己的问题
问题:
我需要将html实体、十进制和十六进制字符引用替换为utf-8等号,就像普通浏览器一样,并将文本转换为utf-8。
问题是,经常有130-150和x82-x9f范围内的引用,正如thirtydot发现的,人们使用ascii文本来处理特殊字符,如emdashes,而php的html实体解码不支持这种字符。
您可能会认为这些无效字符在浏览器中不起作用,但看起来浏览器已经达成了一个无声的、无文档的协议,可以修复这些字符并正确地显示它们。
在尝试修复这些引用时,我还发现实际的字符(如‚)也在使用,这些字符可能是直接从word复制的,并且会导致各种各样的问题,所以我也需要修复它们。
我发现关于编码的大多数答案都没有提到的是,与编码相关的问题的解决方案通常很大程度上取决于所使用的编码。
下面是一个例子:
无效的windows字符‚将与“iso-8859-1”编码文本一起工作,并且invalid windows word characters您应该这样修复它们:

$str = str_replace(chr(151),'--',$str);

它所做的是将windows字符替换为一个安全的ascii选项,但是知道文本将存储在utf-8中,我不想丢失原始字符。
当像这样更改它们时不是一个选项,因为ascii不支持正确的unicode字符:
$str = str_replace(chr(151),chr(8218),$str);

因此,我所做的是首先将字符替换为它的html引用(而$str是“i so-8859-1”编码的:
$str = str_replace(chr(151),'‚'),$str);

然后我改变编码
$str = iconv('ISO-8859-1', 'UTF-8//IGNORE', $str);//convert to UTF-8

最后,我使用我的“html字符参考解码”函数将所有实体和字符引用转换为纯utf-8,该函数主要基于Josh B mentions as per Jukka Korpelas suggestionGumbos,它也修复了错误的windows引用,但只使用&#emdash;来遍历错误的windows字符。
function fix_char_mapping($match){
    if (strtolower($match[1][0]) === "x") {
        $codepoint = intval(substr($match[1], 1), 16);
    } else {
        $codepoint = intval($match[1], 10);
    }
    $mapping = array(8218,402,8222,8230,8224,8225,710,8240,352,8249,338,141,142,143,144,8216,8217,8220,8221,8226,8211,8212,732,8482,353,8250,339,157,158,376); 
    $codepoint = $mapping[$codepoint-130];
    return '&#'.$codepoint.';';
}
function html_character_reference_decode($string, $encoding='UTF-8', $fixMappingBug=true){
    if($fixMappingBug){
        $string = preg_replace_callback('/&#(1[3-5][0-9]|x8[2-9a-f]|x9[0-9a-f]);/i','fix_char_mapping',$string);
    }
    return html_entity_decode($string, ENT_QUOTES, 'UTF-8');
}
header('Content-Type: text; charset=UTF-8');
echo  html_character_reference_decode('dash — and another dash — text ו and more tests נוף ');

因此,如果您的文本是“iso-8859-1”编码的,那么完整的解决方案如下:
<?php
header('Content-Type: text/plain; charset=utf-8');
ini_set("default_charset", 'utf-8');
error_reporting(-1);
$encoding = 'ISO-8859-1';//put encoding here
$str = '&#x9F; &#x9C; bad&#150;string: '.chr(151);//ASCII
if($encoding==='ISO-8859-1'){
//fix bad windows characters
$badchars = array(
'&#130;'=>chr('130'),//',' baseline single quote
'&#131;'=>chr('131'),//'NLG' florin
'&#132;'=>chr('132'),//'"' baseline double quote
'&#133;'=>chr('133'),//'...' ellipsis
'&#134;'=>chr('134'),//'**' dagger (a second footnote)
'&#135;'=>chr('135'),//'***' double dagger (a third footnote)
'&#136;'=>chr('136'),//'^' circumflex accent
'&#137;'=>chr('137'),//'o/oo' permile
'&#138;'=>chr('138'),//'Sh' S Hacek
'&#139;'=>chr('139'),//'<' left single guillemet
'&#140;'=>chr('140'),//'OE' OE ligature
'&#145;'=>chr('145'),//"'" left single quote
'&#146;'=>chr('146'),//"'" right single quote
'&#147;'=>chr('147'),//'"' left double quote
'&#148;'=>chr('148'),//'"' right double quote
'&#149;'=>chr('149'),//'-' bullet
'&#150;'=>chr('150'),//'-' endash
'&#151;'=>chr('151'),//'--' emdash
'&#152;'=>chr('152'),//'~' tilde accent
'&#153;'=>chr('153'),//'(TM)' trademark ligature
'&#154;'=>chr('154'),//'sh' s Hacek
'&#155;'=>chr('155'),//'>' right single guillemet
'&#156;'=>chr('156'),//'oe' oe ligature
'&#159;'=>chr('159'),//'Y' Y Dieresis
);
$str = str_replace(array_values($badchars),array_keys($badchars),$str);
$str = iconv('ISO-8859-1', 'UTF-8//IGNORE', $str);//convert to UTF-8
$str = html_character_reference_decode($str);//fixes bad entities above
echo $str;die;
}

它经过了广泛的测试,看起来很管用。
让我们看看包含错误windows字符的utf-8编码文本的相同情况。
测试是否存在不良字符或“不良格式的utf-8”的一种可靠方法是使用iconv,它很慢,但比在我的测试中使用solution更可靠:
$cleaned = iconv('UTF-8','UTF-8//IGNORE',$str);
if ($cleaned!==$str){
    //contains bad characters, use cleaned version where the bad characters were stripped
    $str = $cleaned;
}

这几乎是我能想到的最好的方法,因为我没有找到合理的方法来查找和替换utf-8文本中的坏windows字符,让我解释一下原因。
使用具有完全有效的Unicode字符<?php echo chr(151);?>和错误的Windows Emdash的字符串。
我不知道utf-8字符串中可能存在哪些坏的windows字符,只知道它们可能存在。
使用chr(151)尝试修复上面的有效emdash字符串中的错误windows字符preg_replace_callback(右双引号),该字符串甚至不包含任何双引号,这将导致一个scrambed字符,起初我认为$str = "—".chr(151);可能不是多字节安全的,并尝试使用str_replace但问题是同样。
php网站和stackoverflow上的评论提到chr(148)是二进制安全的,并且可以很好地处理格式良好的utf-8文本,这是因为utf-8的设计方式。
为什么会破裂
它认为坏的windows字符str_replace由以下位“10010100”组成,而
(emdash字符(preg_match),根据fileformat网站,它由3个字节组成:“11100010:10000000:10010100”
请注意,完全有效的utf-8字符的最后一个字节中的位与坏窗口右双引号中的位匹配,因此mb_eregi_replace只替换最后一个字节,中断utf-8字符。
这个问题发生在许多unicode字符上,例如会扰乱俄语文本中的许多字符。
这不能发生在ascii文本中,因为每个字符总是由一个字节组成。
所以当你得到一个包含任意多字节字符的utf-8字符串时,你就不能再安全地修复坏的windows字符了,我找到的唯一解决办法就是用iconv把它们去掉。
$str = iconv('UTF-8', 'UTF-8//IGNORE', $str);

我唯一能想到的解决办法
尽管您始终可以将包含一个字节坏字符的有效Unicode字符替换为其编码的对应字符,然后替换坏字符,然后解码好字符,从而保留所有内容:)
这样地:
用如下编码替换str_replace
chr(148)
然后用适当的em破折号替换str_replace
然后将11100010:10000000:10010100解码回&#8212;
但是你必须写下每一个多字节字符,其中包含的字节与坏字符相匹配,才能实现这一点。
相关:http://www.fileformat.info/info/unicode/char/2014/index.htm

关于php - 如何将带有HTML实体和无效字符的文本转换为其等效的UTF-8?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9210473/

有关php - 如何将带有HTML实体和无效字符的文本转换为其等效的UTF-8?的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. Ruby 解析字符串 - 2

    我有一个字符串input="maybe(thisis|thatwas)some((nice|ugly)(day|night)|(strange(weather|time)))"Ruby中解析该字符串的最佳方法是什么?我的意思是脚本应该能够像这样构建句子:maybethisissomeuglynightmaybethatwassomenicenightmaybethiswassomestrangetime等等,你明白了......我应该一个字符一个字符地读取字符串并构建一个带有堆栈的状态机来存储括号值以供以后计算,还是有更好的方法?也许为此目的准备了一个开箱即用的库?

  4. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  5. ruby-on-rails - unicode 字符串的长度 - 2

    在我的Rails(2.3,Ruby1.8.7)应用程序中,我需要将字符串截断到一定长度。该字符串是unicode,在控制台中运行测试时,例如'א'.length,我意识到返回了双倍长度。我想要一个与编码无关的长度,以便对unicode字符串或latin1编码字符串进行相同的截断。我已经了解了Ruby的大部分unicode资料,但仍然有些一头雾水。应该如何解决这个问题? 最佳答案 Rails有一个返回多字节字符的mb_chars方法。试试unicode_string.mb_chars.slice(0,50)

  6. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  7. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  8. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  9. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  10. ruby - 使用 ruby​​ 将 HTML 转换为纯文本并维护结构/格式 - 2

    我想将html转换为纯文本。不过,我不想只删除标签,我想智能地保留尽可能多的格式。为插入换行符标签,检测段落并格式化它们等。输入非常简单,通常是格式良好的html(不是整个文档,只是一堆内容,通常没有anchor或图像)。我可以将几个正则表达式放在一起,让我达到80%,但我认为可能有一些现有的解决方案更智能。 最佳答案 首先,不要尝试为此使用正则表达式。很有可能你会想出一个脆弱/脆弱的解决方案,它会随着HTML的变化而崩溃,或者很难管理和维护。您可以使用Nokogiri快速解析HTML并提取文本:require'nokogiri'h

随机推荐