参考:动网论坛形式的无级分类排序
-----------------------------
private void button3_Click(object sender, System.EventArgs e)
{ SqlConnection cnn=new SqlConnection("server=BEAR;user id=sa;initial catalog=123;persist security info=True;");
cnn.Open();

TreeNode node=new TreeNode();
treeView1.Nodes.Add(ShowAllTree(0,cnn,node)); //根节是以PID为0标志的 cnn.Close();
}private TreeNode ShowAllTree(int id,SqlConnection cnn,TreeNode node)
{ string sqlSelect = "SELECT id,pid FROM [1] where pid ='"+id+"'";
SqlDataAdapter da = new SqlDataAdapter(sqlSelect,cnn);
DataTable dt=new DataTable();
da.Fill(dt);

for(int i=0;i<dt.Rows.Count;i++)           
{
TreeNode newNode=new TreeNode(dt.Rows[i][0].ToString());
node.Nodes.Add(newNode); int newID = (int)dt.Rows[i][0];; ShowAllTree(newID,cnn,newNode); //递归调用,将子结点ID当做下一当前结点ID }
dt.Dispose();
da.Dispose(); return node;
}测试数据表1:
id pid
1 0
2 1
3 2
4 3
5 1
6 0
7 0
8 6

解决方案 »

  1.   

    2.用递归生成树的算法和数据库设计
    (1)递归说明
      程序调用自身的编程方法称为递归(recursion)。在树的生成以及图的遍历中,递归用的很多。经典的算法求 n! (求n的阶乘)中,用的就是递归方法。递归算法的优点就是简洁,可扩充性好。但是缺点也很明显:低效。因为递归就是程序不断调用自身,对系统的资源消耗比较大。随着节点的增多,执行效率会变的很低。 
      为了解决树在生成过程中的层树不定的问题,同时也是为了让树的扩展性更好。树的生成使用了递归的方法。生成树的代码一旦写成,可以不改动源代码,生成无限级层次的树。树的结构完全由数据库中表的数据决定。
    (2)数据库设计
      创建一个数据库,设计树图信息表Treetable,表中属性包含Nodeid、Parentid、Nodename、Address等字段(分别用于表示节点的ID、父节点ID、节点名称、链接地址),其它属性根据实际用户需求和设计而定。节点名称Nodename将在树型控件的节点上显示,Nodeid字段保存节点的唯一标识号,Parentid表示当前节点的父节点ID号(例如有两个节点是父子关系,孩子节点的Parentid值就是其父节点的Nodeid),节点号父子相接组成了一个“链表”,表征并记录了树上节点的层次结构。
    表具体设计如下:(简化模型,实际使用的要复杂一些)主键 属性名 类型 长度 可空 属性含义
    是 Nodeid int 6 否 节点ID
    Parentid int 6 否 父节点ID
    Nodename char 50 否 节点名称
    Address char 80 可 链接地址备注:链接地址 主要是用在: 树在框架中使用的环境。链接可以指向其他框架页中的地址或是带不同的参数。
    (3)程序代码
    ――――――――――――递归函数――――――――――――
        ''生成树的函数
        Private Sub inittree(ByRef Nds As Microsoft.Web.UI.WebControls.TreeNodeCollection, ByVal parentId As Integer)
            Dim dv As New DataView()
            Dim dvrow As DataRowView
            Dim tmpNode As Microsoft.Web.UI.WebControls.TreeNode
            ''intId为数值型变量,其作用是记录并传递当前记录的ID,做为它子节点的PARENTID值
            Dim intId As Integer
            dv.Table = mySet.Tables("paybasic")
            ''parentId传递的是 additem函数中的intId.下面语句的作用是找出当前节点的子孩子集合。 
            dv.RowFilter = "parentID=''" & parentId & "''"
            ''如果当前节点有孩子,则遍历所有的孩子,并调用递归函数。
            For Each dvrow In dv
                tmpNode = New Microsoft.Web.UI.WebControls.TreeNode()
                ''为当前节点的各个属性赋值。
                tmpNode.ID = dvrow("nodeID")
                tmpNode.Text = dvrow("nodename")
                tmpNode.NavigateUrl = dvrow("Address")
                intId = dvrow("parentID")
                ''添加一个节点
                Nds.Add(tmpNode)
                ''调用递归函数
                inittree(Nds(Nds.Count - 1).Nodes, intId)
            Next
    End Sub
    ――――――――――――――――调用递归函数――――――――――――――――――
    CreateReaderDataSet()
    inittree(Treepaybasic.Nodes, 999)
    ―――――――――――――――――生成数据集―――――――――――――――――――
        ''生成数据集的函数
        Private Sub CreateReaderDataSet()
            ''在运行时连接,并设置连接属性
            MyConn = New System.Data.OleDb.OleDbConnection("Provider=MSDAORA.1;Data Source=oracle9;User ID=user;Password=****;")
            ''设置SelectCommand命令
            myAdapter.SelectCommand = New System.Data.OleDb.OleDbCommand("select * from treenode", MyConn)
            ''填充数据集
            myAdapter.Fill(mySet, "treenode")
        End Sub
    (1)首先正确的下载和安装ieWebcontrols
    (2)在vs.net中用  [工具箱]-【web窗口】-【自定义工具箱】-【.net框架组件】-
      添加命名空间在
      Microsoft.Web.UI.WebControls中的 treeview
      而不仅仅是
      system.windows.forms 中的 treeview
    与数据库相关的节点操作:
    (1)添加节点
      1.1 节点的树型添加
            //程序的功能是在点击的节点下面添加新节点
            Dim tmpNd3 As New Microsoft.Web.UI.WebControls.TreeNode()
            Dim NdSel As New Microsoft.Web.UI.WebControls.TreeNode()
            ''NdSel 为当前选定的要删除节点,tmpNd3为它的父节点 
            NdSel = Treepaybasic.GetNodeFromIndex(Treepaybasic.SelectedNodeIndex)
            ''要添加的节点的各个属性
            tmpNd3.ID = 111
            tmpNd3.Text = "aaa"
            ''用nodes.add添加节点
            NdSel.Nodes.Add(tmpNd3)
               
        1.2 数据库中的节点添加操作
                //数据库连接语句在此省略
                ''在运行时连接,并设置连接属性
                Dim insertcomm = New System.Data.OleDb.OleDbCommand()
                ''定义存储命令的各种属性
                insertcomm.CommandText = "insert into treebasic(id,parentid,name)      values(''" & pid & "'',''" & parid & "'',''" & nodetext & "'')"
                insertcomm.Connection = MyConn
                ''打开连接、执行命令
                MyConn.Open()
                insertcomm.ExecuteNonQuery()
                MyConn.Close()
      

  2.   

    是呀
    性能低了不合格
    我写的就是
    参考了动网的代码
    minrange(Minrange) :
    能给点代码学习学习吗?
      

  3.   

    楼主不就是要点击姓名的节点,这个人的各级单位显示在窗体的label上吗?
    没有那么难吧,我也写了一个:)private void treeView1_AfterSelect(object sender, System.Windows.Forms.TreeViewEventArgs e)
    {
    string path = null; TreeNode node = this.treeView1.SelectedNode;
    if (node != null)
    {
    path = node.Text;
    while (node.Parent != null )
    {
    node = node.Parent;
    path = node.Text + "\\"+ path;
    }
    }
    this.label1.Text = path;
    }
      

  4.   

    嘿嘿
    先要添加的
    你~~~~
    还没把数据库添加到treeview呢
      

  5.   

    TO: minrange(Minrange) 可不可以贴出来代码来让我学习学习!!!
      

  6.   

    其实以上的程序还是没完成
    真正的数据库里各级单位是无条件添加的
    其中1级单位里的下一级单位是不同的
    按照这个添加下去,就把每一个二级单位添加到了1级单位里面了
    这在程序里面是不允许的
    例如:
    1级单位:a1,a2,a3
    其中二级单位b1是a1的二级单位,不是a2的,所以不能添加到a2的子节点去。
    应该避免这个问题
    如果要避免这个问题
    数据库的检索就变得更慢,
    请高手指点。
      

  7.   

    而且每一个节点的子节点不能重复
    有牵扯到一个group问题