|
過濾xml 數據:
讓我們在假設一下,如果在一個真實的聯系應用程序中,用戶可能不希望在分級的treeview中看到“email”,”city ” 或者“country”,他們可能更希望看到頂級的用戶聯系身份內容,例如Alex, Rebekah, 或者 Justin,既然這樣聯系點的相應的詳細信息(email,city )就處于相鄰的可以編輯的區域,類似的用戶也可能希望通過上下拖拉樹節點來重新排列他們,然而在個體聯系點內部通過treeview控件移動 email 地址或者city 是沒有意義的。人們通常希望用一個分等級的視圖去組織數據,而不簡單的是對他們進行分類,換句話對聯系(contact)進行重新排序分組是很正常的要求,除非city 和country 元素與指定聯系的關聯是很清晰,而且你可能希望單獨的處理他們,而不是把他們進行分組分對。
一個很好的解決方案是當顯示聯系樹時隱藏子節點,例如你可以為email. Address等不想顯示的子元素添加一個特殊的屬性(例如view=”hide”),這樣在組裝樹控件時在組裝方法中設置這個特殊屬性,從而可以忽略任何元素(包括他們的子節點),雖然這樣可以工作,但是改變數據源來適合用戶顯示不是一個很可靠的設計思路
一個更好的思路是在給定的文檔內部為數據客戶定義分級視圖是否可見的結構,你可以通過修改populateTreeControl() 方法以使它支持xpath 例如:
[C#] private void populateTreeControl(System.Xml.XmlNode document, System.Windows.Forms.TreeNodeCollection nodes) { foreach (System.Xml.XmlNode node in document.ChildNodes) { System.Xml.XmlNode expr = node.SelectSingleNode(xpath_filter); if (expr != null) { TreeNode new_child = new TreeNode(expr.Value); nodes.Add(new_child); populateTreeControl(node, new_child.Nodes); } } }
[VB] Private Sub populateTreeControl( _ ByVal document As System.Xml.XmlNode, _ ByVal nodes As System.Windows.Forms.TreeNodeCollection) Dim node As System.Xml.XmlNode For Each node In document.ChildNodes Dim expr As System.Xml.XmlNode = _ node.SelectSingleNode(xpath_filter) If Not (expr Is Nothing) Then Dim new_child As New TreeNode(expr.Value) nodes.Add(new_child) populateTreeControl(node, new_child.Nodes) End If Next End Sub 在類級別范圍上添加下面的行:
[C#] private string xpath_filter = "@id[parent::contacts or parent::contact]";
[VB] Private xpath_filter As String = _ "@id[parent::contacts or parent::contact]"
你可以使用xpath查詢返回的結果來決定是否遞歸調用裝入子節點,這個查詢建立了一個包含規則,讀“Select the id attribute of any 'contacts' or 'contact' element.",同樣你也可以使用一個排除規則去確定你需要拒絕哪一個數據。
attribute::id[not(parent::email or parent::city or parent::country)] 這不是一個通用的解決方案,但是像這樣基于父子關系的過濾比基于沒有限制的節點或者屬性要好很多,當用戶擁有充足的可編輯權限而不妨礙它的層次時,這是一個表達一個xml文檔基本結構的很有效的方法。既然這樣一個簡單的查詢就已經足夠了 ,除非你需要進行更復雜的操作
|