XML严禁 & < ,建议 < > & ' “进行转义

XML中预定义了5个实体引用: < > & ' “

其中,'<' 和 '&' 是非法的。

'<' 会产生错误,因为解析器会把该字符解释为新元素的开始。

'&' 也会产生错误,因为解析器会把该字符解释为字符实体的开始。

其它实体虽然都是合法的,但是把它们替换为实体是个好的习惯。

CDATA区段中值的限制

CDATA区段由 '' 结束.因此,CDATA的值不能包含']]>',否则XML解析器会提前闭合CDATA,导致解析错误.而结尾的']]>'闭合也不能包含空格或者折行.

<![CDATA[ value ]]> ]]> //值包含']]>',非法
<![CDATA[ value ] ] > //结尾的']]>'包含空格,非法
<![CDATA[ value ]]> //合法

SimpleXML类生成带有CDATA区段的值

SimpleXML类如其名字,因为其简单那易用性,得到一部分程序员的青睐,但是其本身却不支持生成带有CDATA区段的值.是不是为了生成CDATA区段的值就要放弃SimpleXML类了呢?

谷歌到一种解决方案

class SimpleXMLExtended extends SimpleXMLElement
{
public function addCData($cdata_text)
{
$node = dom_import_simplexml($this);
$no = $node->ownerDocument;
$node->appendChild($no->createCDATASection($cdata_text));
}
}
$xmlFile = 'config.xml';
// instead of $xml = new SimpleXMLElement('<sites/>');
$xml = new SimpleXMLExtended('<sites/>');
$site = $xml->addChild('site');
// instead of $site->addChild('site', 'Site Title');
$site->title = NULL; // VERY IMPORTANT! We need a node where to append
$site->title->addCData('Site Title');
$site->title->addAttribute('lang', 'en');
$xml->asXML($xmlFile);

此类继承自SimpleXMLElement类,通过添加addCData方法,利用DOMDocument和SimpleXML的互通性,把SimpleXML对象转换成DOM元素对象,利用DOM元素对象节点可以创建CDATA区段值的特点完成操作.

此类虽然解决了SimpleXML不能创建CDATA区段值的缺点,但是代码不够简洁--每次都要在设置值之前都要先将值设置为NULL,而此方法又是必须的,否则会报错.

因此我对此类进行了再次封装,调用起来更加简洁,代码如下:

class SimpleXMLExtended extends SimpleXMLElement
{
public function addCData($data)
{
$node = dom_import_simplexml($this);
$no = $node->ownerDocument;
$node->appendChild($no->createCDATASection($data));
} public function addNode($key,$value=''){
if($value === '') $value = ' ';
$this->$key = NULL;
$this->$key->addCData($value);
}
} //调用更加简洁
$xml = new SimpleXMLElement('<?xml version="1.0"?><DOCUMENT></DOCUMENT>');
$xml->addChild('item');//向XML节点添加一个子节点item
$item->addNode('key','value');//向item节点添加一个键为"key",CDATA区段值为"value"

UTF-8编码下提示非UTF-8字符

XML中编码全是UTF-8,怎么会有非UTF-8字符呢?通过浏览器查看源码,也没发现非法字符.

问大神同事,得到了'没遇到过'的答案.

然后就想着把报错的数据字段从数据库里复制出来,看一下结果.从库中复制到了sublime中,然后看到如下图的'FS'字符

好端端的内容,为啥会产生一个'FS'呢?然后谷歌,终于找到了答案.原来该特殊字符叫'控制字符',想要了解控制字符的同学可以看关于控制字符的定义.

当时解决方法简直简单粗暴,手动删掉,然后回存到数据库:

后来领导也遇到相似的问题,我告诉他是有可能是控制字符造成的,然后他给了我一个函数,可以过滤掉控制字符.函数如下:

function strip_control_characters($string){
return preg_replace('/[\x00-\x1F\x7F-\x9F]/u', '', $string);
}

**PS: 此函数需要编码为UTF-8才可以过滤掉. **

XML读取的字段中包含'-',要用'{}'把字段包裹

$arr = array('a-b-c'=>'abc');
$obj = (object)($arr);
//如何获取'a-b-c'字段的值? echo $obj->a-b-c;//报错
echo $obj->'a-b-c';//报错
echo $obj->{'a-b-c'};//正常

** 此注意点可以总结为PHP在获取对象的key中包含'-',要用'{}'包住 **

SimpleXML的节点值要强转成字符型后再比较

$string = <<<XML
&lt;?xml version='1.0'?&gt;
&lt;document&gt;
&lt;title&gt;Dear Jane&lt;/title&gt;
&lt;from&gt;Joe&lt;/from&gt;
&lt;to&gt;Jane&lt;/to&gt;
&lt;body&gt;
Dear Jane:
I love U,can you be my girlfriend?
yours Joe
&lt;/body&gt;
&lt;/document&gt;
XML; $email = simplexml_load_string($string); //Joe发送邮件,但是忘记收件人是否是女神Jane if($email->to == 'Jane'){
echo 'Good Job';
}else{
echo 'I am in troblem!';
}

得到的答案是'I am in troblem!'

原来'$email->to'得到的值的类型为SimpleXMLElement对象,

非字符串,非数组.

因此,一定要强制转换成'string'类型后在比较.

XML调试

因为SimpleXML类和DOMDocument类都是基于libxml扩展开发的,开发过程中,可以使用libxml_use_internal_errors调试.

此函数有一个参数,默认为false,即禁用用户错误处理.true为开启用户错误处理.

libxml_use_internal_errors(true);
$sxe = simplexml_load_string("<?xml version='1.0'><broken><xml></broken>");
if ($sxe === false) {
echo "Failed loading XML\n";
foreach(libxml_get_errors() as $error) {
echo "\t", $error->message;
}
}

以上代码输出:

Failed loading XML

Blank needed here

parsing XML declaration: '?>' expected

Opening and ending tag mismatch: xml line 1 and broken

Premature end of data in tag broken line 1

通过错误处理,知道原来有两个错误:

  1. 声明少写一个 '?>'

  2. xml标签没有闭合

XML中的值得注意的"坑"的更多相关文章

  1. IDEA中写MyBatis的xml配置文件编译报错的坑

    IDEA中写MyBatis的xml配置文件编译报错的坑 说明:用IDEA编译工具在项目中使用Mybatis框架,编写mybatis-config.xml和Mapper.xml配置文件时,编译项目出现错 ...

  2. Odoo “坑” 系列之 XML中的布尔类型

    在Odoo中试图通过XML方式更新某条Record的值,却意外发现根本不能更新,经查,对于XML中Boolean类型的字段,更新的方式应该采用eval的方式.

  3. Android XML中引用自定义内部类view的四个why

    今天碰到了在XML中应用以内部类形式定义的自定义view,结果遇到了一些坑.虽然通过看了一些前辈写的文章解决了这个问题,但是我看到的几篇都没有完整说清楚why,于是决定做这个总结. 使用自定义内部类v ...

  4. xml解析、写入遇到的坑

    前言 最近在看一个线上xml文件导出的问题,需求如下: 从我们平台导出一个后缀为tmx的术语语料数据(实际内容为xml文件),然后导入到其他第三方平台发现无法导入. 从其他平台导入的tmx文件无法导入 ...

  5. mapper.xml中转义

    1.用转义字符转义 XML转义字符 < < 小于号 > > 大于号 & & 和 &apos; ’ 单引号 " " 双引号 <i ...

  6. xml中的<![CDATA[]]> 简介

    被<![CDATA[]]>这个标记所包含的内容将表示为纯文本,比如<![CDATA[<]]>表示文本内容“<”.  此标记用于xml文档中,我们先来看看使用转义符的 ...

  7. Android中点击按钮获取string.xml中内容并弹窗提示

    场景 AndroidStudio跑起来第一个App时新手遇到的那些坑: https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/103797 ...

  8. xml中的<![CDATA[]]>和转义字符

    被<![CDATA[]]>这个标记所包含的内容将表示为纯文本,比如<![CDATA[<]]>表示文本内容"<". 此标记用于xml文档中,我们先 ...

  9. web.xml中welcome-file-list的作用

    今天尝试使用struts2+ urlrewrite+sitemesh部署项目,结果发现welcome-file-list中定义的欢迎页不起作用: <welcome-file-list> & ...

随机推荐

  1. Eureka 系列(02)Eureka 一致性协议

    目录 Eureka 系列(02)Eureka 一致性协议 0. Spring Cloud 系列目录 - Eureka 篇 1. 服务发现方案对比 1.1 技术选型 1.2 数据模型 2. Eureka ...

  2. js中函数的创建和调用都发生了什么?执行环境,函数作用域链,变量对象

    博客搬迁,给你带来的不便,敬请谅解! http://www.suanliutudousi.com/2017/11/26/js%E4%B8%AD%E5%87%BD%E6%95%B0%E7%9A%84%E ...

  3. el-form-item内容过多,及弹窗框宽度属性show-overflow-tooltip设置

    内容过多: :show-overflow-tooltip=true 宽度属性设置: .el-tooltip__popper{ max-width:30% }

  4. CG-CTF web部分wp

    bin不动了,学学webWEB1,签到1f12,得到flag2,签到2给了输入窗口和密码,但输入后却显示错误,查看源码,发现对输入长度进行了限制,改下长度,得到flag3,md5 collision给 ...

  5. List、Map、Set三个接口存取元素时,各有什么特点

    List接口以特定索引来存取元素,可以有重复元素 Set接口不可以存放重复元素(使用equals方法区分是否重复) Map接口保存的是键值对(key-value-pair)映射,映射关系可以是一对一或 ...

  6. 聊聊动态链接和dl_runtime_resolve

    写在前面 linux下的动态链接相关结构,重新回顾_dl_runtime_resolve的流程以及利用方法 动态链接相关结构 为了高效率的利用内存,多个进程可以共享代码段.程序模块化方便更新维护等,动 ...

  7. C# 连接Excel,获取表格数据,获取多个sheet中的数据,获取多个sheet名

    /// <summary> /// 获取Excel内容. /// </summary> /// <param name="sheetName"> ...

  8. 检查目录下 文件的权限-linux shell脚本

    #!/bin/bash #History: #2019/07/23    Fsq #This Program will check Permissions on dir PATH=/bin:/sbin ...

  9. windows如何在当前路径下快速打开cmd

    直接在打开的文件夹地址栏当中输入cmd即可.

  10. 浅谈HTTP与其工作流程

    一.什么是HTTP协议 HTTP协议(Hyper Text Transfer Protocol)翻译过来是超文本传输协议,也是一种restful风格的协议,在web开发和APP接口开发都很常用. HT ...