最近项目上在做电子病历,使用的是第三方的电子病历组件,该病历组件是利用XML来组织数据的。界面上渲染出来的效果如下图:
XML渲染后的界面
对应的后台数据(已做简化处理)是如下XML格式的,其中的空格部分是用 <Text> </Text>来标记的。
<XTextBody>
<Text>姓名:</Text>
<Text> </Text>
<BackgroundText>姓名</BackgroundText>
<Text> </Text>
<Text> 年龄:</Text>
<Text> </Text>
<BackgroundText>年龄</BackgroundText>
<Text> </Text>
<Text> 性别:</Text>
<Text> </Text>
<BackgroundText>性别</BackgroundText>
<Text> </Text>
<Text>出生日期:</Text>
<Text> </Text>
<BackgroundText>年</BackgroundText>
<Text> </Text>
<Text>年</Text>
<Text> </Text>
<BackgroundText>月</BackgroundText>
<Text> </Text>
<Text>月</Text>
<Text> </Text>
<BackgroundText>日</BackgroundText>
<Text> </Text>
<Text>日</Text>
<Text> </Text>
</XTextBody>
在数据库(SQLServer)中,存储该文本的字段类型是XML类型。
当我们把这一段文本存入数据库并再次加载时,发现空格丢失了
渲染后的效果也就乱掉了,文字都挤到了一起,如下图:
空格丢失,文本挤在一起了
经调查发现,原来是xml:space搞的鬼。
xml:space 属性是一个 XML 定义的属性,用于声明对象元素中重要的空格处理行为。 它有两个可选值:
- default:该属性的默认值,XML解析后将自动删除元素中空格。<text> </text>处理后变成<text />
- preserve:保留元素中的空格。<text> </text>处理后保持不变<text> </text>
如果XML中缺少xml:space这个属性,将会同default行为一样。
知道原因之后,解决起来也就容易了。由于XML是在第三方组件中生成的,短时间内无法修复,那我们就在自己的程序中做处理。在将数据保存入库之前,给根节点加上xml:space="preserve"属性就好了。代码如下:
//保留xml中空白字符
var doc = new XmlDocument { PreserveWhitespace = true };
doc.LoadXml(xml);
doc.DocumentElement.SetAttribute("xml:space", "preserve");
这里再说一个坑,构建XmlDocument对象时,一定要设置属性PreserveWhitespace = true,否则LoadXml后空格又会自动消失啦。因为XmlDocument和数据库里操作是一样的,都是默认缺省xml:space处理的,估计是为了节省空间吧