XmlUtil.cs 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694
  1. using System;
  2. using System.Text;
  3. using System.IO;
  4. using System.Xml;
  5. using System.Reflection;
  6. using System.Collections;
  7. using System.Collections.Generic;
  8. using CommonLang.Property;
  9. using CommonLang.IO;
  10. using CommonLang.Log;
  11. using CommonLang.Concurrent;
  12. namespace CommonLang.Xml
  13. {
  14. public enum XmlSerializableProperty : uint
  15. {
  16. Mark = 0,
  17. IgnoreClone = 0x0001,
  18. NoSerialize = 0x0002,
  19. }
  20. /// <summary>
  21. /// 标记当前Field或者Property是否参与Xml序列化
  22. /// </summary>
  23. [AttributeUsage(AttributeTargets.Property)]
  24. public class XmlSerializableAttribute : System.Attribute
  25. {
  26. private readonly XmlSerializableProperty Prop;
  27. public XmlSerializableAttribute(XmlSerializableProperty attr = XmlSerializableProperty.Mark)
  28. {
  29. this.Prop = attr;
  30. }
  31. public bool IgnoreClone { get { return (Prop & XmlSerializableProperty.IgnoreClone) != 0; } }
  32. public bool NoSerialize { get { return (Prop & XmlSerializableProperty.NoSerialize) != 0; } }
  33. }
  34. /// <summary>
  35. /// Xml序列化与反序列化
  36. /// </summary>
  37. public static class XmlUtil
  38. {
  39. private static Logger s_log;
  40. private static Logger log
  41. {
  42. get
  43. {
  44. if (s_log == null) { s_log = LoggerFactory.GetLogger("XmlUtil"); }
  45. return s_log;
  46. }
  47. }
  48. static public XmlDocument LoadXML(byte[] data)
  49. {
  50. using (MemoryStream ms = new MemoryStream(data))
  51. {
  52. return LoadXML(ms);
  53. }
  54. }
  55. static public XmlDocument LoadXML(Stream input, bool autoDisposeStream = false)
  56. {
  57. try
  58. {
  59. using (XmlReader xml = XmlReader.Create(input))
  60. {
  61. XmlDocument doc = new XmlDocument();
  62. doc.Load(xml);
  63. return doc;
  64. }
  65. }
  66. finally
  67. {
  68. if (autoDisposeStream)
  69. {
  70. input.Close();
  71. input.Dispose();
  72. }
  73. }
  74. }
  75. static public XmlDocument LoadXML(string path)
  76. {
  77. var stream = Resource.LoadDataAsStream(path);
  78. if (stream != null)
  79. {
  80. try
  81. {
  82. return LoadXML(stream);
  83. }
  84. finally
  85. {
  86. stream.Dispose();
  87. }
  88. }
  89. return null;
  90. }
  91. static public void SaveXML(string path, XmlDocument xml)
  92. {
  93. using (FileStream fs = new FileStream(path, FileMode.Create, FileAccess.Write))
  94. {
  95. XmlUtil.SaveXML(fs, xml, false);
  96. }
  97. }
  98. static public string ToString(XmlDocument doc)
  99. {
  100. using (StringWriter sw = new StringWriter())
  101. {
  102. XmlWriterSettings settings = new XmlWriterSettings();
  103. settings.Indent = true;
  104. settings.Encoding = CUtils.UTF8;
  105. using (XmlWriter xml = XmlWriter.Create(sw, settings))
  106. {
  107. doc.Save(xml);
  108. xml.Flush();
  109. }
  110. return sw.ToString();
  111. }
  112. }
  113. static public XmlDocument FromString(string xmltext)
  114. {
  115. XmlDocument doc = new XmlDocument();
  116. doc.LoadXml(xmltext);
  117. return doc;
  118. }
  119. static public bool SaveXML(String path, XmlDocument doc, out string errMessage)
  120. {
  121. try
  122. {
  123. using (FileStream fs = new FileStream(path, FileMode.Create, FileAccess.Write))
  124. {
  125. XmlWriterSettings settings = new XmlWriterSettings();
  126. settings.Indent = true;
  127. settings.Encoding = Encoding.UTF8;
  128. using (XmlWriter xml = XmlWriter.Create(fs, settings))
  129. {
  130. doc.Save(xml);
  131. xml.Flush();
  132. }
  133. errMessage = null;
  134. fs.Close();
  135. }
  136. }
  137. catch(Exception e)
  138. {
  139. errMessage = e.Message;
  140. }
  141. return errMessage == null;
  142. }
  143. static public void SaveXML(Stream output, XmlDocument doc, bool autoDisposeStream)
  144. {
  145. try
  146. {
  147. XmlWriterSettings settings = new XmlWriterSettings();
  148. settings.Indent = true;
  149. settings.Encoding = Encoding.UTF8;
  150. using (XmlWriter xml = XmlWriter.Create(output, settings))
  151. {
  152. doc.Save(xml);
  153. xml.Flush();
  154. }
  155. }
  156. finally
  157. {
  158. if (autoDisposeStream)
  159. {
  160. output.Close();
  161. output.Dispose();
  162. }
  163. }
  164. }
  165. static public string GetTextValue(XmlElement e)
  166. {
  167. if (e.FirstChild != null && e.FirstChild.NodeType == XmlNodeType.Text)
  168. {
  169. return (e.FirstChild as XmlText).Data;
  170. }
  171. else { return null; }
  172. }
  173. static public void SaveToXML(Stream output, object mData, bool autoDisposeStream = false)
  174. {
  175. try
  176. {
  177. Type type = mData.GetType();
  178. XmlDocument doc = XmlUtil.ObjectToXml(mData);
  179. XmlWriterSettings settings = new XmlWriterSettings();
  180. settings.Indent = true;
  181. settings.Encoding = Encoding.UTF8;
  182. using (XmlWriter xml = XmlWriter.Create(output, settings))
  183. {
  184. doc.Save(xml);
  185. xml.Flush();
  186. }
  187. }
  188. finally
  189. {
  190. if (autoDisposeStream)
  191. {
  192. output.Close();
  193. output.Dispose();
  194. }
  195. }
  196. }
  197. //----------------------------------------------------------------------------------------------------------
  198. #region _converter_
  199. static public T XmlToObject<T>(XmlDocument doc, Action<Exception> error = null)
  200. {
  201. object ret = XmlToObject(typeof(T), doc, false, error);
  202. if (ret != null)
  203. {
  204. return (T)ret;
  205. }
  206. return default(T);
  207. }
  208. static public object XmlToObject(XmlDocument doc, Action<Exception> error = null)
  209. {
  210. XmlElement e = (XmlElement)doc.DocumentElement;
  211. Type type = GetTypeFromAttribute(e, "type");
  212. return XmlToObject(type, doc, false, error);
  213. }
  214. static public object XmlToObject(Type type, XmlDocument doc, Action<Exception> error = null)
  215. {
  216. XmlElement e = (XmlElement)doc.DocumentElement;
  217. return XmlToObject(type, doc, false, error);
  218. }
  219. static public object XmlToObject(Type type, XmlDocument doc, bool cloning, Action<Exception> error = null)
  220. {
  221. XmlElement e = (XmlElement)doc.DocumentElement;
  222. object data = ObjectFromXmlElement(e, type, cloning, error);
  223. return data;
  224. }
  225. static public XmlDocument ObjectToXml(object data, string root_name = null)
  226. {
  227. return ObjectToXml(data, root_name, false);
  228. }
  229. static public XmlDocument ObjectToXml(object data, string root_name, bool cloning)
  230. {
  231. Type type = data.GetType();
  232. if (root_name == null)
  233. {
  234. root_name = type.Name;
  235. }
  236. XmlDocument doc = new XmlDocument();
  237. XmlElement e = doc.CreateElement(root_name);
  238. ObjectToXmlElement(e, data, cloning);
  239. doc.AppendChild(e);
  240. return doc;
  241. }
  242. static public T CloneObject<T>(T src)
  243. {
  244. Type type = src.GetType();
  245. if (type.IsPrimitive)
  246. {
  247. return src;
  248. }
  249. else if (type.IsEnum)
  250. {
  251. return src;
  252. }
  253. else if (type.IsAssignableFrom(typeof(string)))
  254. {
  255. return src;
  256. }
  257. else if (type.IsClass || type.IsArray)
  258. {
  259. XmlDocument xml = XmlUtil.ObjectToXml(src, "cloning", true);
  260. T obj = (T)XmlUtil.XmlToObject(type, xml, true);
  261. return obj;
  262. }
  263. return src;
  264. }
  265. #endregion
  266. //----------------------------------------------------------------------------------------------------------
  267. #region _internal_
  268. static private void ObjectFieldsFromXMLElement(XmlElement data_element, object data, bool cloning, Action<Exception> error)
  269. {
  270. Type type = data.GetType();
  271. foreach (XmlNode fe in data_element.ChildNodes)
  272. {
  273. XmlElement fee = fe as XmlElement;
  274. FieldInfo fii = type.GetField(fe.Name);
  275. if (fii != null)
  276. {
  277. object fd = ObjectFromXmlElement(fee, fii.FieldType, cloning, error);
  278. fii.SetValue(data, fd);
  279. }
  280. }
  281. }
  282. static private void ObjectToXmlElement(XmlElement data_element, object data, bool cloning)
  283. {
  284. if (data != null)
  285. {
  286. Type type = data.GetType();
  287. if (type.IsPrimitive)
  288. {
  289. data_element.InnerText = Parser.ObjectToString(data);
  290. }
  291. else if (type.IsEnum)
  292. {
  293. data_element.InnerText = Parser.ObjectToString(data);
  294. }
  295. else if (type.IsAssignableFrom(typeof(string)))
  296. {
  297. data_element.InnerText = Parser.ObjectToString(data);
  298. }
  299. else if (type.IsArray)
  300. {
  301. ObjectToXMLElementArray(data_element, (Array)data, cloning);
  302. }
  303. else if (type.IsClass)
  304. {
  305. XmlDocument doc = data_element.OwnerDocument;
  306. if (type.GetInterface(typeof(IDictionary).Name) != null)
  307. {
  308. ObjectToXMLElementMap(data_element, (IDictionary)data, cloning);
  309. }
  310. else if (type.GetInterface(typeof(IList).Name) != null)
  311. {
  312. ObjectToXMLElementList(data_element, (IList)data, cloning);
  313. }
  314. else
  315. {
  316. SetTypeToAttribute(data_element, "type", type);
  317. foreach (FieldInfo field in PropertyUtil.SortFields(type.GetFields()))
  318. {
  319. if (!field.IsStatic)
  320. {
  321. object fd = field.GetValue(data);
  322. if (fd != null)
  323. {
  324. XmlElement fe = doc.CreateElement(field.Name);
  325. ObjectToXmlElement(fe, fd, cloning);
  326. data_element.AppendChild(fe);
  327. }
  328. }
  329. }
  330. foreach (PropertyInfo property in PropertyUtil.SortProperties(type.GetProperties()))
  331. {
  332. XmlSerializableAttribute attr = PropertyUtil.GetAttribute<XmlSerializableAttribute>(property);
  333. if (attr != null)
  334. {
  335. if (attr.NoSerialize)
  336. {
  337. }
  338. else if (cloning && attr.IgnoreClone)
  339. {
  340. }
  341. else
  342. {
  343. object fd = property.GetValue(data, null);
  344. if (fd != null)
  345. {
  346. XmlElement fe = doc.CreateElement("property." + property.Name);
  347. ObjectToXmlElement(fe, fd, cloning);
  348. data_element.AppendChild(fe);
  349. }
  350. }
  351. }
  352. }
  353. }
  354. }
  355. }
  356. }
  357. static private object ObjectFromXmlElement(XmlElement data_element, Type type, bool cloning, Action<Exception> error)
  358. {
  359. try
  360. {
  361. Type s_type = GetTypeFromAttribute(data_element, "type", type);
  362. if (s_type != null)
  363. {
  364. type = s_type;
  365. }
  366. if (type.IsPrimitive)
  367. {
  368. return Parser.StringToObject(data_element.InnerText, type);
  369. }
  370. else if (type.IsEnum)
  371. {
  372. return Parser.StringToObject(data_element.InnerText, type);
  373. }
  374. else if (type.IsAssignableFrom(typeof(string)))
  375. {
  376. return data_element.InnerText;
  377. }
  378. else if (type.IsArray)
  379. {
  380. return ObjectFromXMLElementArray(data_element, type, cloning, error);
  381. }
  382. else if (type.IsClass)
  383. {
  384. if (type.GetInterface(typeof(IDictionary).Name) != null)
  385. {
  386. return ObjectFromXMLElementMap(data_element, type, cloning, error);
  387. }
  388. else if (type.GetInterface(typeof(IList).Name) != null)
  389. {
  390. return ObjectFromXMLElementList(data_element, type, cloning, error);
  391. }
  392. else
  393. {
  394. try
  395. {
  396. object data = ReflectionUtil.CreateInstance(type);
  397. foreach (XmlNode fe in data_element.ChildNodes)
  398. {
  399. XmlElement fee = fe as XmlElement;
  400. if (fe.Name.StartsWith("property."))
  401. {
  402. PropertyInfo property = type.GetProperty(fe.Name.Substring("property.".Length));
  403. if (property != null)
  404. {
  405. XmlSerializableAttribute attr = PropertyUtil.GetAttribute<XmlSerializableAttribute>(property);
  406. if (attr != null)
  407. {
  408. if (attr.NoSerialize)
  409. {
  410. }
  411. else if (cloning && attr.IgnoreClone)
  412. {
  413. }
  414. else
  415. {
  416. object fd = ObjectFromXmlElement(fee, property.PropertyType, cloning, error);
  417. if (fd != null)
  418. {
  419. property.SetValue(data, fd, null);
  420. }
  421. }
  422. }
  423. }
  424. }
  425. else
  426. {
  427. FieldInfo fii = type.GetField(fe.Name);
  428. if (fii != null)
  429. {
  430. object fd = ObjectFromXmlElement(fee, fii.FieldType, cloning, error);
  431. if (fd != null)
  432. {
  433. fii.SetValue(data, fd);
  434. }
  435. }
  436. }
  437. }
  438. return data;
  439. }
  440. catch (Exception err)
  441. {
  442. log.Warn(err.Message, err);
  443. if (error != null) error(err);
  444. }
  445. }
  446. }
  447. }
  448. catch (Exception err)
  449. {
  450. log.Warn(err.Message, err);
  451. if (error != null) error(err);
  452. }
  453. return null;
  454. }
  455. static private void ObjectToXMLElementArray(XmlElement data_element, Array array, bool cloning)
  456. {
  457. XmlDocument doc = data_element.OwnerDocument;
  458. Type type = array.GetType();
  459. int rank = type.GetArrayRank();
  460. int[] ranges = new int[rank];
  461. for (int i = 0; i < rank; i++)
  462. {
  463. ranges[i] = array.GetLength(i);
  464. }
  465. SetTypeToAttribute(data_element, "type", type);
  466. SetTypeToAttribute(data_element, "element_type", type.GetElementType());
  467. data_element.SetAttribute("rank", rank + "");
  468. data_element.SetAttribute("ranges", Parser.ObjectToString(ranges));
  469. foreach (object k in array)
  470. {
  471. XmlElement ei = doc.CreateElement("element");
  472. ObjectToXmlElement(ei, k, cloning);
  473. data_element.AppendChild(ei);
  474. }
  475. }
  476. static private Array ObjectFromXMLElementArray(XmlElement data_element, Type type, bool cloning, Action<Exception> error)
  477. {
  478. int rank = int.Parse(data_element.GetAttribute("rank"));
  479. Type ctype = GetTypeFromAttribute(data_element, "type");
  480. Type etype = GetTypeFromAttribute(data_element, "element_type");
  481. int[] ranges = (int[])Parser.StringToObject(data_element.GetAttribute("ranges"), typeof(int[]));
  482. Array array = Array.CreateInstance(etype, ranges);
  483. int total_index = 0;
  484. foreach (XmlNode fe in data_element.ChildNodes)
  485. {
  486. XmlElement fee = fe as XmlElement;
  487. object fdd = ObjectFromXmlElement(fee, etype, cloning, error);
  488. int[] indices = CUtils.GetArrayRankIndex(ranges, total_index);
  489. array.SetValue(fdd, indices);
  490. total_index++;
  491. }
  492. return array;
  493. }
  494. static private void ObjectToXMLElementMap(XmlElement data_element, IDictionary map, bool cloning)
  495. {
  496. XmlDocument doc = data_element.OwnerDocument;
  497. Type type = map.GetType();
  498. if (type.IsGenericType)
  499. {
  500. SetTypeToAttribute(data_element, "key_type", type.GetGenericArguments()[0]);
  501. SetTypeToAttribute(data_element, "value_type", type.GetGenericArguments()[1]);
  502. }
  503. foreach (object k in map.Keys)
  504. {
  505. object v = map[k];
  506. XmlElement epair = doc.CreateElement("element");
  507. XmlElement ek = doc.CreateElement("key");
  508. XmlElement ev = doc.CreateElement("value");
  509. if (!type.IsGenericType)
  510. {
  511. SetTypeToAttribute(ek, "type", k.GetType());
  512. SetTypeToAttribute(ev, "type", v.GetType());
  513. }
  514. ObjectToXmlElement(ek, k, cloning);
  515. ObjectToXmlElement(ev, v, cloning);
  516. epair.AppendChild(ek);
  517. epair.AppendChild(ev);
  518. data_element.AppendChild(epair);
  519. }
  520. }
  521. static private IDictionary ObjectFromXMLElementMap(XmlElement data_element, Type type, bool cloning, Action<Exception> error)
  522. {
  523. Type k_type = GetTypeFromAttribute(data_element, "key_type");
  524. Type v_type = GetTypeFromAttribute(data_element, "value_type");
  525. IDictionary map = (IDictionary)Activator.CreateInstance(type);
  526. foreach (XmlNode fe in data_element.ChildNodes)
  527. {
  528. XmlElement epair = fe as XmlElement;
  529. XmlElement ek = epair.ChildNodes[0] as XmlElement;
  530. XmlElement ev = epair.ChildNodes[1] as XmlElement;
  531. k_type = GetTypeFromAttribute(ek, "type", k_type);
  532. v_type = GetTypeFromAttribute(ev, "type", v_type);
  533. object k = ObjectFromXmlElement(ek, k_type, cloning, error);
  534. object v = ObjectFromXmlElement(ev, v_type, cloning, error);
  535. map.Add(k, v);
  536. }
  537. return map;
  538. }
  539. static private void ObjectToXMLElementList(XmlElement data_element, IList list, bool cloning)
  540. {
  541. XmlDocument doc = data_element.OwnerDocument;
  542. Type type = list.GetType();
  543. if (type.IsGenericType)
  544. {
  545. SetTypeToAttribute(data_element, "element_type", type.GetGenericArguments()[0]);
  546. }
  547. foreach (object k in list)
  548. {
  549. XmlElement ei = doc.CreateElement("element");
  550. if (!type.IsGenericType)
  551. {
  552. SetTypeToAttribute(ei, "type", k.GetType());
  553. }
  554. ObjectToXmlElement(ei, k, cloning);
  555. data_element.AppendChild(ei);
  556. }
  557. }
  558. static private IList ObjectFromXMLElementList(XmlElement data_element, Type type, bool cloning, Action<Exception> error)
  559. {
  560. Type element_type = GetTypeFromAttribute(data_element, "element_type");
  561. IList list = (IList)Activator.CreateInstance(type);
  562. foreach (XmlNode fe in data_element.ChildNodes)
  563. {
  564. XmlElement ei = fe as XmlElement;
  565. element_type = GetTypeFromAttribute(ei, "type", element_type);
  566. Object fd = ObjectFromXmlElement(ei, element_type, cloning, error);
  567. list.Add(fd);
  568. }
  569. return list;
  570. }
  571. static private void SetTypeToAttribute(XmlElement e, string name, Type type)
  572. {
  573. e.SetAttribute(name, type.FullName);
  574. }
  575. static private Type GetTypeFromAttribute(XmlElement e, string name)
  576. {
  577. string vtype = e.GetAttribute(name);
  578. if (vtype != null)
  579. {
  580. Type ret = ReflectionUtil.GetType(vtype);
  581. if (ret != null)
  582. {
  583. return ret;
  584. }
  585. }
  586. return null;
  587. }
  588. static private Type GetTypeFromAttribute(XmlElement e, string name, Type defaultType)
  589. {
  590. string vtype = e.GetAttribute(name);
  591. if (!String.IsNullOrEmpty(vtype))
  592. {
  593. Type ret = ReflectionUtil.GetType(vtype);
  594. if (ret != null)
  595. {
  596. return ret;
  597. }
  598. }
  599. return defaultType;
  600. }
  601. #endregion
  602. //----------------------------------------------------------------------------------------------------------
  603. #region _utils_
  604. public static bool TryGetAttribute(XmlNode e, string key, out string value, bool NotNull = true)
  605. {
  606. XmlAttribute attr = e.Attributes[key];
  607. if (attr != null)
  608. {
  609. if (NotNull && string.IsNullOrEmpty(attr.Value))
  610. {
  611. value = null;
  612. return false;
  613. }
  614. value = attr.Value;
  615. return true;
  616. }
  617. value = null;
  618. return false;
  619. }
  620. public static string GetAttribute(XmlNode e, string key, bool NotNull = true)
  621. {
  622. XmlAttribute attr = e.Attributes[key];
  623. if (attr != null)
  624. {
  625. if (NotNull && string.IsNullOrEmpty(attr.Value))
  626. {
  627. return null;
  628. }
  629. return attr.Value;
  630. }
  631. return null;
  632. }
  633. public static XmlNode FindChild(XmlNode e, string childName)
  634. {
  635. foreach (XmlNode cc in e.ChildNodes)
  636. {
  637. if (cc.Name == childName)
  638. {
  639. return cc;
  640. }
  641. }
  642. return null;
  643. }
  644. #endregion
  645. //----------------------------------------------------------------------------------------------------
  646. }
  647. }