Modeling.cs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Collections;
  5. using System.Reflection;
  6. using CommonLang.Xml;
  7. namespace CommonLang.Property.Modeling
  8. {
  9. public class UmlNode : IComparable<UmlNode>
  10. {
  11. public string Name { get; private set; }
  12. public HashMap<string, object> Attributes { get; private set; }
  13. public UmlNode(string name)
  14. {
  15. this.Name = name;
  16. this.Attributes = new HashMap<string, object>();
  17. }
  18. public UmlNode ParentNode { get { return mParent; } }
  19. public IEnumerable<UmlNode> ChildNodes { get { return mChildNodes; } }
  20. private UmlNode mParent;
  21. private List<UmlNode> mChildNodes = new List<UmlNode>();
  22. private HashMap<string, UmlNode> mChildNodesMap = new HashMap<string, UmlNode>();
  23. internal void AppendChild(UmlNode child)
  24. {
  25. if (child.mParent != null)
  26. {
  27. throw new Exception(string.Format("Child already have a parent : child={0} parent={1}", child.Name, child.mParent.Name));
  28. }
  29. child.mParent = this;
  30. this.mChildNodes.Add(child);
  31. this.mChildNodes.Sort();
  32. this.mChildNodesMap.Add(child.Name, child);
  33. }
  34. internal void ClearChilds()
  35. {
  36. foreach (UmlNode node in mChildNodes)
  37. {
  38. node.mParent = null;
  39. }
  40. this.mChildNodes.Clear();
  41. this.mChildNodesMap.Clear();
  42. }
  43. public UmlNode GetChild(string key)
  44. {
  45. return this.mChildNodesMap.Get(key);
  46. }
  47. public virtual int CompareTo(UmlNode other)
  48. {
  49. return this.Name.CompareTo(other.Name);
  50. }
  51. }
  52. public class UmlDocument : UmlNode
  53. {
  54. public object Data;
  55. public readonly UmlValueNode DocumentElement;
  56. private TableClassAttribute mDataTypeDesc;
  57. public TableClassAttribute DataTypeDesc { get { return mDataTypeDesc; } }
  58. public UmlDocument(object data)
  59. : base(data.GetType().Name)
  60. {
  61. this.mDataTypeDesc = PropertyUtil.GetAttribute<TableClassAttribute>(data.GetType());
  62. this.Data = data;
  63. this.DocumentElement = CreateValueNode("Document", data.GetType(), null);
  64. this.DocumentElement.BindValue(data, null);
  65. this.AppendChild(DocumentElement);
  66. }
  67. static public UmlValueNode CreateValueNode(string name, Type type, object owner)
  68. {
  69. if (type.IsArray)
  70. {
  71. return new UmlArrayNode(name, type, owner);
  72. }
  73. else if (type.IsPrimitive)
  74. {
  75. return new UmlLeafValueNode(name, type, owner);
  76. }
  77. else if (type.IsEnum)
  78. {
  79. return new UmlLeafValueNode(name, type, owner);
  80. }
  81. else if (type.IsAssignableFrom(typeof(string)))
  82. {
  83. return new UmlLeafValueNode(name, type, owner);
  84. }
  85. else if (type.IsClass)
  86. {
  87. if (type.GetInterface(typeof(IDictionary).Name) != null)
  88. {
  89. return new UmlMapNode(name, type, owner);
  90. }
  91. else if (type.GetInterface(typeof(IList).Name) != null)
  92. {
  93. return new UmlListNode(name, type, owner);
  94. }
  95. else
  96. {
  97. return new UmlFieldsValueNode(name, type, owner);
  98. }
  99. }
  100. else
  101. {
  102. return new UmlFieldsValueNode(name, type, owner);
  103. }
  104. }
  105. }
  106. public abstract class UmlValueNode : UmlNode
  107. {
  108. public UmlValueNode ParentValueNode { get { return ParentNode as UmlValueNode; } }
  109. public object OwnerIndexer { get { return mOwnerIndexer; } }
  110. public object OwnerObject { get { return mOwnerObject; } }
  111. public Type ValueType { get { return mValueType; } }
  112. public object Value { get { return mValue; } }
  113. public DescAttribute FieldDesc
  114. {
  115. get
  116. {
  117. if (OwnerIndexer is FieldInfo)
  118. {
  119. return PropertyUtil.GetAttribute<DescAttribute>(OwnerIndexer as FieldInfo);
  120. }
  121. return null;
  122. }
  123. }
  124. public bool IsRoot
  125. {
  126. get { return this.OwnerObject == null; }
  127. }
  128. public bool IsLeaf
  129. {
  130. get
  131. {
  132. if (mValueType.IsPrimitive)
  133. {
  134. return true;
  135. }
  136. if (mValueType.IsEnum)
  137. {
  138. return true;
  139. }
  140. if (mValueType.IsAssignableFrom(typeof(string)))
  141. {
  142. return true;
  143. }
  144. if (mValueType.IsClass)
  145. {
  146. return false;
  147. }
  148. return false;
  149. }
  150. }
  151. private readonly Type mValueType;
  152. private readonly object mOwnerObject;
  153. private object mOwnerIndexer;
  154. private object mValue;
  155. public UmlValueNode(string name, Type type, object owner)
  156. : base(name)
  157. {
  158. this.mValueType = type;
  159. this.mOwnerObject = owner;
  160. }
  161. internal void BindValue(object value, object key)
  162. {
  163. this.mValue = value;
  164. this.mOwnerIndexer = key;
  165. this.ClearChilds();
  166. this.GenChilds();
  167. }
  168. protected abstract void GenChilds();
  169. /// <summary>
  170. /// 设置当前节点值(此操作 将改变当前节点结构)
  171. /// </summary>
  172. /// <param name="value"></param>
  173. public void SetValue(object value)
  174. {
  175. if (ParentNode is UmlValueNode)
  176. {
  177. UmlValueNode parent = ParentNode as UmlValueNode;
  178. parent.SetFieldValue(this, value);
  179. }
  180. }
  181. /// <summary>
  182. /// 设置当前节点值(此操作 将改变当前节点结构)
  183. /// </summary>
  184. public abstract void SetFieldValue(UmlValueNode child, object value);
  185. }
  186. public class UmlFieldsValueNode : UmlValueNode
  187. {
  188. public UmlFieldsValueNode(string name, Type type, object owner)
  189. : base(name, type, owner)
  190. {
  191. }
  192. protected override void GenChilds()
  193. {
  194. if (Value != null && !IsLeaf)
  195. {
  196. foreach (FieldInfo field in ValueType.GetFields())
  197. {
  198. if (!field.IsStatic && !field.IsLiteral && !field.IsInitOnly && field.IsPublic)
  199. {
  200. Type ft = field.FieldType;
  201. object fv = field.GetValue(Value);
  202. if (fv != null) { ft = fv.GetType(); }
  203. UmlValueNode fe = UmlDocument.CreateValueNode(field.Name, ft, Value);
  204. fe.BindValue(fv, field);
  205. AppendChild(fe);
  206. }
  207. }
  208. }
  209. }
  210. public override void SetFieldValue(UmlValueNode child, object value)
  211. {
  212. child = GetChild(child.Name) as UmlValueNode;
  213. if (child != null)
  214. {
  215. FieldInfo field = child.OwnerIndexer as FieldInfo;
  216. field.SetValue(this.Value, value);
  217. child.BindValue(value, field);
  218. }
  219. }
  220. }
  221. public class UmlLeafValueNode : UmlValueNode
  222. {
  223. public UmlLeafValueNode(string name, Type type, object owner)
  224. : base(name, type, owner)
  225. {
  226. }
  227. protected override void GenChilds()
  228. {
  229. }
  230. public override void SetFieldValue(UmlValueNode child, object value)
  231. {
  232. }
  233. }
  234. public class UmlArrayNode : UmlValueNode
  235. {
  236. readonly public Type ElementType;
  237. readonly public int Rank;
  238. public ListAttribute ListDesc
  239. {
  240. get
  241. {
  242. if (OwnerIndexer is FieldInfo)
  243. {
  244. return PropertyUtil.GetAttribute<ListAttribute>(OwnerIndexer as FieldInfo);
  245. }
  246. return null;
  247. }
  248. }
  249. public Array ArrayValue
  250. {
  251. get
  252. {
  253. if (Value != null) { return (Array)Value; } return null;
  254. }
  255. }
  256. public UmlArrayNode(string name, Type type, object owner)
  257. : base(name, type, owner)
  258. {
  259. this.ElementType = ValueType.GetElementType();
  260. this.Rank = ValueType.GetArrayRank();
  261. }
  262. protected override void GenChilds()
  263. {
  264. if (Value != null)
  265. {
  266. Array array = (Array)Value;
  267. int i = 0;
  268. foreach (object v in array)
  269. {
  270. if (v != null)
  271. {
  272. Type vtype = v.GetType();
  273. UmlValueNode ei = UmlDocument.CreateValueNode(i.ToString(), vtype, Value);
  274. ei.BindValue(v, i);
  275. AppendChild(ei);
  276. }
  277. i++;
  278. }
  279. }
  280. }
  281. public override void SetFieldValue(UmlValueNode child, object value)
  282. {
  283. child = GetChild(child.Name) as UmlValueNode;
  284. if (child != null)
  285. {
  286. int i = (int)child.OwnerIndexer;
  287. Array array = (Array)this.Value;
  288. array.SetValue(value, i);
  289. child.BindValue(value, i);
  290. }
  291. }
  292. }
  293. public class UmlListNode : UmlValueNode
  294. {
  295. readonly public Type GenericElementType;
  296. public ListAttribute ListDesc
  297. {
  298. get
  299. {
  300. if (OwnerIndexer is FieldInfo)
  301. {
  302. return PropertyUtil.GetAttribute<ListAttribute>(OwnerIndexer as FieldInfo);
  303. }
  304. return null;
  305. }
  306. }
  307. public IList ListValue
  308. {
  309. get
  310. {
  311. if (Value != null) { return (IList)Value; } return null;
  312. }
  313. }
  314. public UmlListNode(string name, Type type, object owner)
  315. : base(name, type, owner)
  316. {
  317. if (ValueType.IsGenericType)
  318. {
  319. GenericElementType = ValueType.GetGenericArguments()[0];
  320. }
  321. }
  322. protected override void GenChilds()
  323. {
  324. if (Value != null)
  325. {
  326. IList list = (IList)Value;
  327. int i = 0;
  328. foreach (object v in list)
  329. {
  330. if (v != null)
  331. {
  332. Type vtype = v.GetType();
  333. UmlValueNode ei = UmlDocument.CreateValueNode(i.ToString(), vtype, Value);
  334. ei.BindValue(v, i);
  335. AppendChild(ei);
  336. }
  337. i++;
  338. }
  339. }
  340. }
  341. public override void SetFieldValue(UmlValueNode child, object value)
  342. {
  343. child = GetChild(child.Name) as UmlValueNode;
  344. if (child != null)
  345. {
  346. int i = (int)child.OwnerIndexer;
  347. IList list = (IList)this.Value;
  348. list[i] = value;
  349. child.BindValue(value, i);
  350. }
  351. }
  352. }
  353. public class UmlMapNode : UmlValueNode
  354. {
  355. readonly public Type GenericKeyType;
  356. readonly public Type GenericValueType;
  357. public IDictionary MapValue
  358. {
  359. get
  360. {
  361. if (Value != null) { return (IDictionary)Value; } return null;
  362. }
  363. }
  364. public UmlMapNode(string name, Type type, object owner)
  365. : base(name, type, owner)
  366. {
  367. if (ValueType.IsGenericType)
  368. {
  369. GenericKeyType = ValueType.GetGenericArguments()[0];
  370. GenericValueType = ValueType.GetGenericArguments()[1];
  371. }
  372. else
  373. {
  374. GenericKeyType = null;
  375. GenericValueType = null;
  376. }
  377. }
  378. protected override void GenChilds()
  379. {
  380. if (Value != null)
  381. {
  382. IDictionary map = (IDictionary)Value;
  383. foreach (object k in map.Keys)
  384. {
  385. object v = map[k];
  386. if (v != null)
  387. {
  388. Type vtype = v.GetType();
  389. UmlValueNode ei = UmlDocument.CreateValueNode(k.ToString(), vtype, this.Value);
  390. ei.BindValue(v, k);
  391. AppendChild(ei);
  392. }
  393. }
  394. }
  395. }
  396. public override void SetFieldValue(UmlValueNode child, object value)
  397. {
  398. child = GetChild(child.Name) as UmlValueNode;
  399. if (child != null)
  400. {
  401. IDictionary map = (IDictionary)this.Value;
  402. map[child.OwnerIndexer] = value;
  403. child.BindValue(value, child.OwnerIndexer);
  404. }
  405. }
  406. }
  407. //---------------------------------------------------------------------------------
  408. //---------------------------------------------------------------------------------
  409. public static class UmlUtils
  410. {
  411. static class UmlListSameElements
  412. {
  413. /// <summary>
  414. /// 保留结构相同部分
  415. /// </summary>
  416. /// <param name="list"></param>
  417. /// <returns></returns>
  418. static public void ListSameElements(UmlDocument[] list)
  419. {
  420. }
  421. }
  422. //---------------------------------------------------------------------------------
  423. /// <summary>
  424. /// 将两个结构内部配平,
  425. /// 比如A对象内的一个数组为2个,B对象内部数组为4个,那么调用后,A对象内的数组将补齐为4个
  426. /// </summary>
  427. /// <param name="src"></param>
  428. /// <param name="dst"></param>
  429. /// <returns>是否发生变化</returns>
  430. static public bool StructEquationBalancer(object src, object dst)
  431. {
  432. return UmlBalancer.StructEquationBalancer(src, dst);
  433. }
  434. static class UmlBalancer
  435. {
  436. static public bool StructEquationBalancer(object src, object dst)
  437. {
  438. if (src == dst)
  439. {
  440. return false;
  441. }
  442. if (src.GetType() != dst.GetType())
  443. {
  444. return false;
  445. }
  446. bool changed = false;
  447. foreach (FieldInfo fi in src.GetType().GetFields())
  448. {
  449. if (!fi.IsStatic && !fi.IsLiteral && !fi.IsInitOnly && fi.IsPublic)
  450. {
  451. Type type = fi.FieldType;
  452. if (type.IsPrimitive)
  453. {
  454. }
  455. else if (type.IsEnum)
  456. {
  457. }
  458. else if (type.IsAssignableFrom(typeof(string)))
  459. {
  460. }
  461. else if (type.IsArray)
  462. {
  463. Array sfva = fi.GetValue(src) as Array;
  464. Array dfva = fi.GetValue(dst) as Array;
  465. if (BalanceArray(ref sfva, ref dfva))
  466. {
  467. fi.SetValue(src, sfva);
  468. fi.SetValue(dst, dfva);
  469. changed = true;
  470. }
  471. }
  472. else if (type.IsClass)
  473. {
  474. object sfv = fi.GetValue(src);
  475. object dfv = fi.GetValue(dst);
  476. if (sfv == null && dfv != null)
  477. {
  478. // 其中一个不为空
  479. sfv = XmlUtil.CloneObject(dfv);
  480. fi.SetValue(src, sfv);
  481. changed = true;
  482. }
  483. else if (sfv != null && dfv == null)
  484. {
  485. // 其中一个不为空
  486. dfv = XmlUtil.CloneObject(sfv);
  487. fi.SetValue(dst, dfv);
  488. changed = true;
  489. }
  490. else if (sfv == null && dfv == null)
  491. {
  492. // 两个都为空
  493. }
  494. else if (type.GetInterface(typeof(IDictionary).Name) != null)
  495. {
  496. if (BalanceMap((IDictionary)sfv, (IDictionary)dfv))
  497. {
  498. changed = true;
  499. }
  500. }
  501. else if (type.GetInterface(typeof(IList).Name) != null)
  502. {
  503. if (BalanceList((IList)sfv, (IList)dfv))
  504. {
  505. changed = true;
  506. }
  507. }
  508. else
  509. {
  510. if (StructEquationBalancer(sfv, dfv))
  511. {
  512. changed = true;
  513. }
  514. }
  515. }
  516. }
  517. }
  518. return changed;
  519. }
  520. //---------------------------------------------------------------------------------
  521. private static bool BalanceMap(IDictionary src, IDictionary dst)
  522. {
  523. return false;
  524. }
  525. /// <summary>
  526. /// 将两个数组配平,少的自动补齐
  527. /// </summary>
  528. /// <param name="src"></param>
  529. /// <param name="dst"></param>
  530. /// <returns></returns>
  531. private static bool BalanceList(IList src, IList dst)
  532. {
  533. bool changed = false;
  534. if (src.Count != dst.Count)
  535. {
  536. if (src.Count < dst.Count)
  537. {
  538. for (int i = src.Count; i < dst.Count; i++)
  539. {
  540. src.Add(XmlUtil.CloneObject(dst[i]));
  541. }
  542. }
  543. else
  544. {
  545. for (int i = dst.Count; i < src.Count; i++)
  546. {
  547. dst.Add(XmlUtil.CloneObject(src[i]));
  548. }
  549. }
  550. changed = true;
  551. }
  552. for (int i = 0; i < src.Count; i++)
  553. {
  554. object ssi = src[i];
  555. object ddi = dst[i];
  556. if (StructEquationBalancer(ssi, ddi))
  557. {
  558. changed = true;
  559. }
  560. }
  561. return changed;
  562. }
  563. /// <summary>
  564. /// 将两个数组配平,少的自动补齐
  565. /// </summary>
  566. /// <param name="src"></param>
  567. /// <param name="dst"></param>
  568. /// <returns></returns>
  569. private static bool BalanceArray(ref Array src, ref Array dst)
  570. {
  571. if (src == null && dst != null)
  572. {
  573. // 其中一个不为空
  574. src = XmlUtil.CloneObject(dst);
  575. return true;
  576. }
  577. else if (src != null && dst == null)
  578. {
  579. // 其中一个不为空
  580. dst = XmlUtil.CloneObject(src);
  581. return true;
  582. }
  583. else if (src == null && dst == null)
  584. {
  585. // 两个都为空
  586. return false;
  587. }
  588. bool changed = false;
  589. ArrayList lis_s = new ArrayList(src);
  590. ArrayList lis_d = new ArrayList(dst);
  591. if (src.Length != dst.Length)
  592. {
  593. if (lis_s.Count < lis_d.Count)
  594. {
  595. int[] ranges = CUtils.GetArrayRanges(dst);
  596. Array array = Array.CreateInstance(dst.GetType(), ranges);
  597. FillArray(lis_d, lis_s, array, ranges);
  598. src = array;
  599. }
  600. else
  601. {
  602. int[] ranges = CUtils.GetArrayRanges(src);
  603. Array array = Array.CreateInstance(src.GetType(), ranges);
  604. FillArray(lis_s, lis_d, array, ranges);
  605. dst = array;
  606. }
  607. changed = true;
  608. }
  609. for (int i = 0; i < src.Length; i++)
  610. {
  611. object ssi = lis_s[i];
  612. object ddi = lis_d[i];
  613. if (StructEquationBalancer(ssi, ddi))
  614. {
  615. changed = true;
  616. }
  617. }
  618. return changed;
  619. }
  620. /// <summary>
  621. /// 将ret按照more标准配平
  622. /// </summary>
  623. /// <param name="more"></param>
  624. /// <param name="less"></param>
  625. /// <param name="ret"></param>
  626. /// <param name="ranges"></param>
  627. private static void FillArray(ArrayList more, ArrayList less, Array ret, int[] ranges)
  628. {
  629. for (int i = less.Count; i < more.Count; i++)
  630. {
  631. less.Add(XmlUtil.CloneObject(more[i]));
  632. }
  633. int total_index = 0;
  634. foreach (object fe in less)
  635. {
  636. int[] indices = CUtils.GetArrayRankIndex(ranges, total_index);
  637. ret.SetValue(fe, indices);
  638. total_index++;
  639. }
  640. }
  641. }
  642. //---------------------------------------------------------------------------------
  643. /// <summary>
  644. /// 深度判断结构是否一致
  645. /// </summary>
  646. /// <param name="r0"></param>
  647. /// <param name="r1"></param>
  648. /// <returns></returns>
  649. static public bool StructEquals(List<UmlNode> r0, List<UmlNode> r1)
  650. {
  651. if (r0.Count != r1.Count)
  652. {
  653. return false;
  654. }
  655. for (int i = r0.Count - 1; i >= 0; --i)
  656. {
  657. if (r0[i].GetType() != r1[i].GetType())
  658. {
  659. return false;
  660. }
  661. if (r0[i] is UmlValueNode)
  662. {
  663. UmlValueNode r0_i = r0[i] as UmlValueNode;
  664. UmlValueNode r1_i = r1[i] as UmlValueNode;
  665. if (r0_i.OwnerIndexer != r1_i.OwnerIndexer)
  666. {
  667. return false;
  668. }
  669. }
  670. if (!string.Equals(r0[i].Name, r1[i].Name))
  671. {
  672. return false;
  673. }
  674. }
  675. return true;
  676. }
  677. //---------------------------------------------------------------------------------
  678. static public List<UmlNode> ListElements(UmlDocument root)
  679. {
  680. List<UmlNode> array = new List<UmlNode>();
  681. ListElements(root.DocumentElement, array);
  682. return array;
  683. }
  684. static public void ListElements(UmlNode data_element, List<UmlNode> array)
  685. {
  686. array.Add(data_element);
  687. foreach (UmlNode fe in data_element.ChildNodes)
  688. {
  689. ListElements(fe, array);
  690. }
  691. }
  692. //---------------------------------------------------------------------------------
  693. }
  694. }