如何将字节数组转换为xml
本文关键字:转换 xml 数组 字节数 字节 | 更新日期: 2023-09-27 17:58:30
我有一个这样的字节流:
byte[] response =
{
69, 90, 69, 45, 88, 77, 76, 45, 77, 115, 103, 48, 50, 60, 77, 101, 115, 115, 97, 103, 101, 62, 13, 10, 32,
32, 60, 72, 101, 97, 100, 101, 114, 62, 13, 10, 32, 32, 32, 32, 60, 77, 101, 115, 115, 97, 103, 101, 68,
97, 116, 101, 62, 50, 48, 49, 48, 48, 51, 50, 52, 60, 47, 77, 101, 115, 115, 97, 103, 101, 68, 97, 116,
101, 62, 13, 10, 32, 32, 32, 32, 60, 77, 101, 115, 115, 97, 103, 101, 84, 105, 109, 101, 62, 49, 57, 50,
56, 48, 54, 60, 47, 77, 101, 115, 115, 97, 103, 101, 84, 105, 109, 101, 62, 13, 10, 32, 32, 60, 47, 72,
101, 97, 100, 101, 114, 62, 13, 10, 32, 32, 60, 66, 111, 100, 121, 62, 13, 10, 32, 32, 32, 32, 60, 84,
114, 97, 110, 115, 97, 99, 116, 105, 111, 110, 73, 68, 62, 51, 51, 50, 53, 50, 55, 60, 47, 84, 114, 97,
110, 115, 97, 99, 116, 105, 111, 110, 73, 68, 62, 13, 10, 32, 32, 32, 32, 60, 84, 114, 97, 110, 115, 97,
99, 116, 105, 111, 110, 78, 117, 109, 98, 101, 114, 62, 49, 50, 49, 48, 52, 55, 48, 60, 47, 84, 114, 97,
110, 115, 97, 99, 116, 105, 111, 110, 78, 117, 109, 98, 101, 114, 62, 13, 10, 32, 32, 32, 32, 60, 80,
104, 111, 110, 101, 78, 117, 109, 98, 101, 114, 62, 54, 51, 57, 57, 57, 57, 57, 57, 57, 57, 57, 57, 60,
47, 80, 104, 111, 110, 101, 78, 117, 109, 98, 101, 114, 62, 13, 10, 32, 32, 32, 32, 60, 65, 109, 111,
117, 110, 116, 62, 48, 48, 48, 48, 48, 48, 50, 53, 48, 48, 60, 47, 65, 109, 111, 117, 110, 116, 62, 13,
10, 32, 32, 32, 32, 60, 82, 101, 115, 117, 108, 116, 62, 48, 51, 60, 47, 82, 101, 115, 117, 108, 116, 62,
13, 10, 32, 32, 60, 47, 66, 111, 100, 121, 62, 13, 10, 60, 47, 77, 101, 115, 115, 97, 103, 101, 62
};
我想从中提取xml。尝试了以下操作:
XmlDocument doc2 = new XmlDocument();
MemoryStream ms = new MemoryStream(response);
doc2.Load(ms);
但有一个例外,比如:
System.Xml.dll 中发生类型为"System.Xml.XmlException"的未处理异常
附加信息:根级别的数据无效。第1行,位置1。
我对这个xml东西真的很陌生,doc2.load()方法应该怎么做?它会创建任何我以后可以读取的xml文件吗?它只是内存中的集合吗?
问题是byte []
数组表示一个开头有一些非XML字符的字符串。如果我做
string s;
using (var ms = new MemoryStream(response))
using (var reader = new StreamReader(ms))
{
s = reader.ReadToEnd();
Console.WriteLine(s);
}
我看到
EZE-XML-Msg02<Message>
<Header>
<MessageDate>20100324</MessageDate>
<MessageTime>192806</MessageTime>
</Header>
<Body>
<TransactionID>332527</TransactionID>
<TransactionNumber>1210470</TransactionNumber>
<PhoneNumber>639999999999</PhoneNumber>
<Amount>0000002500</Amount>
<Result>03</Result>
</Body>
</Message>
必须首先移除EZE-XML-Msg02
。要做到这一点,最好的方法是首先不要将其存储在XML中。但是,如果您无法阻止它被包含在XML中,您可以执行以下操作:
XmlDocument doc2 = new XmlDocument();
using (var ms = new MemoryStream(response))
using (var reader = new StreamReader(ms))
{
while (!reader.EndOfStream && reader.Peek() > -1 && (char)reader.Peek() != '<')
reader.Read();
if (!reader.EndOfStream)
doc2.Load(reader);
}
这是因为它是无效的XML。您的完整字符串如下:
EZE-XML-Msg02<Message>
<Header>
<MessageDate>20100324</MessageDate>
<MessageTime>192806</MessageTime>
</Header>
<Body>
<TransactionID>332527</TransactionID>
<TransactionNumber>1210470</TransactionNumber>
<PhoneNumber>639999999999</PhoneNumber>
<Amount>0000002500</Amount>
<Result>03</Result>
</Body>
</Message>
无关的"EZE-XML-Msg02"导致了这个问题。你要么想修剪掉这个部分(你可以做字符串。如果你事先知道这个部分的长度,就删除它,要么做字符串。拆分(','),如果你不知道,就用它来查找长度)。或者,如果可能的话,最好从一开始就阻止它进入XML。
在这一点上有一些旁注。首先,你的解码方法肯定会起作用,但你可能想把MemoryStream放在一个"使用"块中。另一种解码方法如下:
string str = ASCIIEncoding.ASCII.GetString(response);
它会为你解码字符串。你可以做
doc2.LoadXml(str);
它会为你加载文档。
此时,XmlDocument对象本身将只是一个内存中的集合。如果您愿意,可以通过调用doc2.save(filename)将其显式保存为文件。
还要注意的是,有些人更喜欢LINQ版本(XDocument)而不是XML Document,这样他们就可以针对它运行LINQ查询。XmlDocument对象支持XPath查询,但效率有些低,因为它不能静态编译查询,必须在运行时对其进行解析。不过,你使用哪一种取决于你的具体应用程序。