朋友今天问我一个面试题,我看了看,给大家分享一下

    本人才疏学浅,望大家多给意见,有更好的做法大加分享分享
 
下面是题目:
 
已知表table_department中有两个字段,分别为d_id,d_name。d_id记录的是部门编码, d_name记录的是部门名称,各部门的组织方式如下:
A为顶级部门,A部门的下级部门使用AA、BA、CA……表示
AA的下级部门使用AAA、BAA、CAA……表示
BA的夏季部门使用ABA、BBA、CBA……表示
以此类推。
新建一个应用程序,写一个页面或窗体,讲table_department表中的数据,按树状排列显示,如下所示:
A总经办
-AA生产部
--AAA保修部
--BAA非保部
-BA物流部
--ABA物流一部
--BBA物流二部
--CBA物流三部
-CA市场部
--ACA市场拓展
--BCA营销部
---ABCA电器营销部
---BBCA电子营销部
…………
…………
加分项:
表table_department用XML实现
 
===================================================
 
这个问题我觉得会有比较多的解法,我这里暂且考虑面试的问题,所以给出的解法不会考虑太多严谨的东西
首先建库建表添加数据
 
SQL脚本为
=================================================
 
  
  
  
  
  1. -- 创建数据库 
  2. if db_id('testdb'is not null 
  3.     drop database testdb; 
  4. go 
  5. create database testdb 
  6. on 
  7.     name='testdb'
  8.     filename='e:\db\testdb.mdf' 
  9. log on 
  10.     name='testdb_log'
  11.     filename='e:\db\testdb_log.ldf' 
  12. ); 
  13.  
  14. use testdb; 
  15. --创建数据表 
  16. if object_id('table_department''U'is null 
  17. create table table_department 
  18.     d_id varchar(10), 
  19.     d_name nvarchar(50) 
  20. ); 
  21. go 
  22. -- 添加数据 
  23. insert into table_department(d_id, d_name) values('A''总经办'); 
  24. insert into table_department(d_id, d_name) values('AA''生产部'); 
  25. insert into table_department(d_id, d_name) values('BA''物流部'); 
  26. insert into table_department(d_id, d_name) values('CA''市场部'); 
  27. insert into table_department(d_id, d_name) values('AAA''保修部'); 
  28. insert into table_department(d_id, d_name) values('BAA''非保部'); 
  29. insert into table_department(d_id, d_name) values('ABA''物流一部'); 
  30. insert into table_department(d_id, d_name) values('BBA''物流二部'); 
  31. insert into table_department(d_id, d_name) values('CBA''物流三部'); 
  32. insert into table_department(d_id, d_name) values('ACA''市场拓展'); 
  33. insert into table_department(d_id, d_name) values('BCA''经营部'); 
  34. insert into table_department(d_id, d_name) values('ABCA''电器经营部'); 
  35. insert into table_department(d_id, d_name) values('BBCA''电子经营部'); 
========================================
 
第一种解法
 
也是最简单傻瓜式的解法,使用ADO.Net读取数据. 并将读到的数据,根据d_id字段的数据创建TreeView节点,并加载数据
观察树节点的规律,每个节点只有d_id的现实,只有最后一个节点现实完整的d_id和d_name,并且每个d_id的字符表示一个层次结构
因此可以写一个方法,根据包含d_id和d_name的字符串创建节点和添加数据
简单步骤:
1、 首先该方法要往TreeView添加数据,因此该方法一定要有一个TreeNode参数(鉴于根节点只有一个,可以将A添加为根节点,或直接就将"公司"作为根节点)
2、 观察d_id的字符串,根节点在最右边,子节点在左边(估计是为了故意设计的面试题,这样不好排序)
实际这个很简单,将d_id字符串转换成字符数组,从后往左遍历,并在TreeNode中创建节点,如果节点存在就不用创建
3、 因此方法原型可以定义为:
  
  
  
  
  1. private void ShowFromString(string d_id, string id, string d_name, TreeNode tn) 
  2.             { 
  3.                 // 实现代码 
  4.                 // d_id创建结构使用 
  5.                 // id记录部门id号 
  6.                 // d_name记录部门名称 
  7.                 // tn表示当前节点 
  8.             } 
4、 接下来看方法如何实现
由于TreeNode是有层次显示的,所以这里使用递归最为容易(循环感觉也可以实现)
4.1 首先将d_id编程字符数组,并得到最后一个字符,这个顶级节点
  
  
  
  
  1. char[] chs = d_id.ToCharArray(); 
  2.                 string nodeStr = chs[chs.Length - 1].ToString(); 
4.2 在TreeNode中检索是否存在这个节点. 检索存在就是看TreeNode的子节点中是否有Tag与nodeStr匹配的(这里可使用Linq,不过既然简单用最原始的)
写一个方法,由于部门的名字是不会重复的,所以这么写
 
  
  
  
  
  1. private bool IsExist(string nodeText, TreeNode tn) 
  2.     bool isTrue = false
  3.     for(int i = 0; i < tn.Nodes.Count; i++) 
  4.     { 
  5.         if(tn.Nodes[i].Tag as string == nodeText) 
  6.         { 
  7.             isTrue = true
  8.             break
  9.         } 
  10.     } 
  11.     return isTrue; 
该方法只要找到TreeNode直接子节点中存在与给定字符串相同的节点就返回true,否则返回false
4.3 判断是否存在节点,如果不存在就创建,如果存在就得到这个节点
这里需要注意的是,所有节点的逻辑结构均有Tag属性来确认,而Text属性最终使用d_id与d_name替换,因此这里是一个临时的值
 
  
  
  
  
  1. TreeNode tnObj = null
  2. if(!IsExist(nodeStr, tn)) 
  3.     tnObj = tn.Nodes.Add(nodeStr); 
  4.     tnObj.Tag = nodeStr; 
  5. else 
  6.     // 得到这个节点 
4.4 考虑如果存在就得到该节点,但是不要写循环一次了,太麻烦,因此修改IsExist方法
 
  
  
  
  
  1. private bool IsExist(string nodeText, TreeNode tn, out TreeNode tnObj) 
  2.     tnObj = null
  3.     bool isTrue = false
  4.     for(int i = 0; i < tn.Nodes.Count; i++) 
  5.     { 
  6.         if(tn.Nodes[i].Tag as string == nodeText) 
  7.         { 
  8.             isTrue = true
  9.             tnObj = tn.Nodes[i];    // 将找到的节点直接返回 
  10.             break
  11.         } 
  12.     } 
  13.     return isTrue; 
因此4.3步的代码可以改为
 
  
  
  
  
  1. if(!IsExist(nodeStr, tn, out tnObj)) 
  2.     tnObj = tn.Nodes.Add(nodeStr); 
  3.     tnObj.Tag = nodeStr; 
这个方法的思路来自int.TryParse方法,如果找到了,那么返回true,那么tnObj中就有了该节点
如果没有找到那么创建一个,反正tnObj中就有当前子节点
4.5 这里应该判断是不是最后一个节点,如果是最后一个节点那么就应该将id和d_name加到Text属性上
由于使用递归完成,因此再次调用这个方法的时候,会将存储d_id的char数组最后一个字符去掉
因此使用chs.Length == 1即可判断是否为最后一个节点
  
  
  
  
  1. if(chs.Length == 1) 
  2.             { 
  3.                 // 将当前节点即为结束节点 
  4.                 tnObj.Text = string.Format("{0} {1}", id, d_name); 
  5.             } 
  6.             else 
  7.             { 
  8.                 // 如果不是最终节点,则递归 
  9.                 ShowFromString(new string(chs, 0, chs.Length - 1), id, d_name, tnObj); 
  10.             } 
5、 整合一下方法
 
  
  
  
  
  1. private void ShowFromString(string d_id, string id, string d_name, TreeNode tn) 
  2.     char[] chs = d_id.ToCharArray(); 
  3.     string nodeStr = chs[chs.Length - 1].ToString(); 
  4.     TreeNode tnObj = null
  5.     if(!IsExist(nodeStr, tn, out tnObj)) 
  6.     { 
  7.         tnObj = tn.Nodes.Add(nodeStr); 
  8.         tnObj.Tag = nodeStr; 
  9.     } 
  10.     if(chs.Length == 1) 
  11.     { 
  12.         tnObj.Text = string.Format("{0} {1}", id, d_name); 
  13.     } 
  14.     else 
  15.     { 
  16.         ShowFromString(new string(chs, 0, chs.Length - 1), id, d_name, tnObj); 
  17.     } 
  18. private bool IsExist(string nodeText, TreeNode tn, out TreeNode tnObj) 
  19.     tnObj = null
  20.     bool isTrue = false
  21.     for(int i = 0; i < tn.Nodes.Count; i++) 
  22.     { 
  23.         if(tn.Nodes[i].Tag as string == nodeText) 
  24.         { 
  25.             isTrue = true
  26.             tnObj = tn.Nodes[i];    // 将找到的节点直接返回 
  27.             break
  28.         } 
  29.     } 
  30.     return isTrue; 
6、 添加窗体的Load事件,并添加代码
 
  
  
  
  
  1. private void Form1_Load(object sender, EventAges e) 
  2.     // 添加根节点公司, 就是在公司下面添加节点 
  3.     TreeNode tn = tvCompany.Nodes.Add("公司"); 
  4.     // 处理数据库,读数据 
  5.     using(SqlConnection conn = new SqlConnection(@"server=.\sqlexpress;database=testdb;integrated security=true")) 
  6.     { 
  7.         using(SqlCommand cmd = new SqlCommand("select d_id, d_name from table_department", conn)) 
  8.         { 
  9.             conn.Open(); 
  10.             using(SqlDataReader reader = cmd.ExecuteReader()) 
  11.             { 
  12.                 if(reader.HasRow) 
  13.                 { 
  14.                     while(reader.Read()) 
  15.                     { 
  16.                         string d_id = reader.GetString(0); 
  17.                         string d_name = reader.GetString(1); 
  18.                         ShowFromString(d_id, d_id, d_name, tn); 
  19.                     } 
  20.                 } 
  21.             } 
  22.         } 
  23.     } 
7、 最后要用XML存储,递归遍历节点,创建XML数据,就像遍历文件夹一样
7.1 添加递归方法
 
  
  
  
  
  1. private void GetXML(TreeNode tn, XElement ele) 
  2. {  
  3.     // 得到tn下的数据,并加到ele中 
  4.     for (int i = 0; i < tn.Nodes.Count; i++) 
  5.     { 
  6.         // 创建对应节点 
  7.         XElement ele1 = new XElement(tn.Nodes[i].Text.Replace(" ""_")); // 由于XML中节点名中不允许有空格,所以去掉 
  8.         ele.Add(ele1); 
  9.  
  10.         // 递归 
  11.         GetXML(tn.Nodes[i], ele1); 
  12.     } 
7.2 添加按钮事件
 
  
  
  
  
  1. private void createXML_Click(object sender, EventArgs e) 
  2.     XDocument xDoc = new XDocument(); 
  3.     xDoc.Add(new XElement("Company")); 
  4.  
  5.     GetXML(tvCompany.Nodes[0], xDoc.Root); 
  6.  
  7.     xDoc.Save("company.xml"); 
 
 
第二种方法
第一种方法比较简单,关键在于如何处理d_id结构而已,而且顺序读取和创建
实际上TreeNode与XML结构一致,是可以同样处理的,也就是说先从数据库中取出数据,生成XML数据,在递归遍历XML数据创建TreeNode
1、 从数据库中读取数据,并创建XML文件
树形结构有一个特点,就是每一个节点只允许有一个父节点和一个子节点,所以可以从数据库中取出所有数据,得到所有数据的节点片段数据
在根据一定算法将节点连起来
1.1 添加一个方法,该方法完成从数据库中读取数据,并得到XML集合(数组也行,个人比较喜欢集合)
 
  
  
  
  
  1. private List<XElement> GetElementByDatabase() 
  2.     // 代码 
1.2 读取数据库,创建XML集合
 
  
  
  
  
  1. private List<XElement> GetElementByDatabase() 
  2.     List<XElement> list = new List<XElement>(); 
  3.     using (SqlConnection conn = new SqlConnection(@"server=.\sqlexpress;database=testdb;integrated security=true")) 
  4.     { 
  5.         using (SqlCommand cmd = new SqlCommand("select d_id, d_name from table_department", conn)) 
  6.         { 
  7.             conn.Open(); 
  8.             using (SqlDataReader reader = cmd.ExecuteReader()) 
  9.             { 
  10.                 if (reader.HasRows) 
  11.                 { 
  12.                     while (reader.Read()) 
  13.                     { 
  14.                         string d_id = reader[0].ToString(); 
  15.                         string d_name = reader[1].ToString(); 
  16.                         // 开始生成XML数据 
  17.                         list.Add(new XElement("department"
  18.                                                 new XAttribute("d_id", d_id), 
  19.                                                 new XAttribute("d_name", d_name) 
  20.                                 )); 
  21.                     } 
  22.                 } 
  23.             } 
  24.         } 
  25.     } 
  26.     return list; 
2、 处理XML片段集合的结构,这个结构没有构成树状结构,因此写一个方法将这个XML片段集合变成一个XML树片段
这里算法有很多,也可以使用Linq查询,但是我不打算详细描述算法,因为有些比较抽象
这里用一个不一定最快,但是很直观的算法
2.1 添加一个方法
 
  
  
  
  
  1. public XElement GetXMLTree(List<XElement> listXML) 
  2.     // 代码 
2.2 了解到XML每一个节点至多只有一个父节点和子节点,因此只要将处理好节点的去掉即可
同时每一个节点都是通过d_id分层次,而这个层次很有规律,父节点刚好比子节点多一个字符,也就是说
父节点的d_id与子节点的d_id.Substring(1)相同
所以就可以从最长的节点开始找,依次为每一个节点找父节点即可
 
  
  
  
  
  1. public XElement GetXMLTree(List<XElement> listXML) 
  2.     // 先为listXML降序排序,因为d_id越长,节点越深 
  3.     listXML.Sort((XElement x1, XElement x2) => { return x2.Attribute("d_id").Value.Length - x1.Attribute("d_id").Value.Length; }); 
  4.     // 从左开始为每一个节点找父节点,很显然,最长的节点最深 
  5.     // 一旦找到父节点,添加进去,就可以将该节点从集合中移除 
  6.     for (int i = 0; i < listXML.Count; i++) 
  7.     { 
  8.         XElement curr = listXML[i]; 
  9.         for (int j = i + 1; j < listXML.Count; j++) 
  10.         { 
  11.             // 判断是否为父子关系 
  12.             if (curr.Attribute("d_id").Value.Substring(1) == listXML[j].Attribute("d_id").Value) 
  13.             { 
  14.                 listXML[j].Add(curr); 
  15.             } 
  16.         } 
  17.     } 
  18.     return listXML[listXML.Count - 1]; 
3、 XML结构有了,那么就可以保存该数据了
另外遍历XML结构,加载到TreeView控件中
 
  
  
  
  
  1. private void ShowTreeNode(TreeNode tn, XElement ele) 
  2.     foreach (XElement item in ele.Elements()) 
  3.     { 
  4.         TreeNode tn1 = tn.Nodes.Add(string.Format("{0} {1}", item.Attribute("d_id").Value, item.Attribute("d_name").Value)); 
  5.         if (item.HasElements) 
  6.         { 
  7.             ShowTreeNode(tn1, item); 
  8.         } 
  9.     } 
4、 添加Load方法
 
  
  
  
  
  1. XElement element = null// 记录要保存的XML数据 
  2. private void Form1_Load(object sender, EventArgs e) 
  3.     List<XElement> xelements = GetElementByDatabase(); 
  4.  
  5.     element = GetXMLTree(xelements); 
  6.  
  7.     ShowTreeNode(tvCompany.Nodes.Add("公司"), element); 
5、 添加保存XML的代码(添加XElement字段)
 
  
  
  
  
  1. private void btnSave_Click(object sender, EventArgs e) 
  2.     XDocument xDoc = new XDocument(element); 
  3.     xDoc.Save("xml.xml"); 
  4.     MessageBox.Show("OK"); 
 
 
第三种方法
写了第二种方法就不太想写第三种方法了,介绍一下基本思想吧
为数据表创建一个对象模型,但是多出一个字段,就是记录反序的d_id
那么就可以利用排序等手段创建对象集合
同时解析每一个字符创建TreeView节点了
 
=================================================
 
好了,就给出成型的两个算法吧!如果有时间在慢慢看. 本人见识有限,还请大家多提意见,如果有更好的思路,借鉴一下啊!!!
 
 
2012年10月10日晚
 
 
 
 
 
 
 
 
 
 
 
 
 

你可能感兴趣的:(xml,数据库,treeview)