zoukankan      html  css  js  c++  java
  • C# 数据结构 之 树

    树(Tree)是n(n>=0)个相同类型的数据元素的有限集合。树中的数据元素叫结点(Node)。n=0的树称为空树(Empty Tree);对于n>0的任意非空树T有:

    1.有且只有一个特殊的结点称为树的根(Root)结点,根没有前驱结点。

    2.若n>1,则除根结点外,其余结点被分成了m(m>0)个互不相交的集合T1,T2,T3,...Tm,其中每个集合Ti(1<=i<=m)本身又是一棵树。树T1,T2,...Tm称为这棵树的子树(Subtree)。

    由树的定义可知,树的定义是递归的,用树来定义树,因此,树的许多运算都使用过了递归。

    树的相关术语

    结点(Node)。表示树中的数据元素,由数据项和数据元素之间的关系组成。

    结点的度。结点所拥有的子树的个数。

    树的度。树中各结点度的最大值。

    叶子结点。度为0的结点,也叫终端结点。

    其他术语参考二叉树。

    用多重链表表示法存储树

    每个结点指针域的个数等于树的度数。

    Tree
    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    
    namespace DataStructure
    {
    
        public interface ITree<T>
        {
            T Root();                                //求树的根结点
            T Parent(T t);                           //求结点t的双亲结点
            T Child(T t, int i);                     //求结点t的第i个子结点
            T RightSibling(T t);                     //求结点t第一个右边兄弟结点
            bool Insert(T s, T t, int i);            //将树s加入树中作为结点t的第i颗子树
            T Delete(T t, int i);                    //删除结点t的第i颗子树
            void Traverse(int TraverseType);         //按某种方式遍历树
            void Clear();                            //清空树
            bool IsEmpty();                          //判断是否为空
            int GetDepth(T t);                          //求树的深度
        }
    
    
        /// <summary>
        /// 循环顺序队列
        /// </summary>
        /// <typeparam name="T"></typeparam>
        class CSeqQueue<T>
        {
            private int maxsize;       //循环顺序队列的容量
            private T[] data;          //数组,用于存储循环顺序队列中的数据元素
            private int front;         //指示最近一个已经离开队列的元素所占有的位置 循环顺序队列的对头
            private int rear;          //指示最近一个进入队列的元素的位置           循环顺序队列的队尾
    
            public T this[int index]
            {
                get { return data[index]; }
                set { data[index] = value; }
            }
    
            //容量属性
            public int Maxsize
            {
                get { return maxsize; }
                set { maxsize = value; }
            }
    
            //对头指示器属性
            public int Front
            {
                get { return front; }
                set { front = value; }
            }
    
            //队尾指示器属性
            public int Rear
            {
                get { return rear; }
                set { rear = value; }
            }
    
            public CSeqQueue()
            {
    
            }
    
            public CSeqQueue(int size)
            {
                data = new T[size];
                maxsize = size;
                front = rear = -1;
            }
    
            //判断循环顺序队列是否为满
            public bool IsFull()
            {
                if ((front == -1 && rear == maxsize - 1) || (rear + 1) % maxsize == front)
                    return true;
                else
                    return false;
            }
    
            //清空循环顺序列表
            public void Clear()
            {
                front = rear = -1;
            }
    
            //判断循环顺序队列是否为空
            public bool IsEmpty()
            {
                if (front == rear)
                    return true;
                else
                    return false;
            }
    
            //入队操作
            public void EnQueue(T elem)
            {
                if (IsFull())
                {
                    Console.WriteLine("Queue is Full !");
                    return;
                }
                rear = (rear + 1) % maxsize;
                data[rear] = elem;
            }
    
            //出队操作
            public T DeQueue()
            {
                if (IsEmpty())
                {
                    Console.WriteLine("Queue is Empty !");
                    return default(T);
                }
                front = (front + 1) % maxsize;
                return data[front];
            }
    
            //获取对头数据元素
            public T GetFront()
            {
                if (IsEmpty())
                {
                    Console.WriteLine("Queue is Empty !");
                    return default(T);
                }
                return data[(front + 1) % maxsize];//front从-1开始
            }
    
            //求循环顺序队列的长度
            public int GetLength()
            {
                return (rear - front + maxsize) % maxsize;
            }
        }
    
        /// <summary>
        /// 树的多链表结点类
        /// </summary>
        /// <typeparam name="T"></typeparam>
        class MLNode<T>
        {
            private T data;                   //存储结点的数据
            private MLNode<T>[] childs;       //存储子结点的位置
    
            public MLNode(int max)
            {
                childs = new MLNode<T>[max];
                for (int i = 0; i < childs.Length; i++)
                {
                    childs[i] = null;
                }
            }
    
            public T Data
            {
                get { return data; }
                set { data = value; }
            }
    
            public MLNode<T>[] Childs
            {
                get { return childs; }
                set { childs = value; }
            }
        }
    
    
    
        class MLTree<T> : ITree<MLNode<T>>
        {
            private MLNode<T> head;
    
            public MLNode<T> Head
            {
                get { return head; }
                set { head = value; }
            }
    
            public MLTree()
            {
                head = null;
            }
    
            public MLTree(MLNode<T> node)
            {
                head = node;
            }
    
            //求树的根结点
            public MLNode<T> Root()
            {
                return head;
            }
    
            public void Clear()
            {
                head = null;
            }
    
            //待测试!!!
            public int GetDepth(MLNode <T> root)
            {
                int len;
                if (root == null)
                {
                    return 0;
                }
                for (int i = 0; i < root.Childs.Length; i++)
                {
                    if (root.Childs[i] != null)
                    {
                        len = GetDepth(root.Childs[i]);
                        return len + 1;
                    }
                }
                return 0;
            }
    
            public bool IsEmpty()
            {
                return head == null;
            }
    
            //求结点t的双亲结点,如果t的双亲结点存在,返回双亲结点,否则返回空
            //按层序遍历的算法进行查找
            public MLNode<T> Parent(MLNode<T> t)
            {
                MLNode<T> temp = head;
                if (IsEmpty() || t == null) return null;
                if (temp.Data.Equals(t.Data)) return null;
                CSeqQueue<MLNode<T>> queue = new CSeqQueue<MLNode<T>>(50);
                queue.EnQueue(temp);
                while (!queue.IsEmpty())
                {
                    temp = (MLNode<T>)queue.DeQueue();
                    for (int i = 0; i < temp.Childs.Length; i++)
                    {
                        if (temp.Childs[i] != null)
                        {
                            if (temp.Childs[i].Data.Equals(t.Data))
                            {
                                return temp;
                            }
                            else
                            {
                                queue.EnQueue(temp.Childs[i]);
                            }
                        }
                    }
                }
                return null;
            }
    
            //求结点t的第i个子结点。如果存在,返回第i个子结点,否则返回空
            //i=0时,表示求第一个子节点
            public MLNode<T> Child(MLNode<T> t, int i)
            {
                if (t != null && i < t.Childs.Length)
                {
                    return t.Childs[i];
                }
                else
                {
                    return null;
                }
            }
    
            //求结点t第一个右边兄弟结点,如果存在,返回第一个右边兄弟结点,否则返回空
            public MLNode<T> RightSibling(MLNode<T> t)
            {
                MLNode<T> pt = Parent(t);
                if (pt != null)
                {
                    int i = FindRank(t);
                    return Child(pt, i + 1);
                }
                else
                {
                    return null;
                }
            }
    
            //查找结点t在兄弟中的排行,成功时返回位置,否则返回-1
            private int FindRank(MLNode<T> t)
            {
                MLNode<T> pt = Parent(t);
                for (int i = 0; i < pt.Childs.Length; i++)
                {
                    MLNode<T> temp = pt.Childs[i];
                    if (temp != null && temp.Data.Equals(t.Data))
                    {
                        return i;
                    }
                }
                return -1;
            }
    
            //查找在树中的结点t,成功是返回t的位置,否则返回null
            private MLNode<T> FindNode(MLNode<T> t)
            {
                if (head.Data.Equals(t.Data)) return head;
                MLNode<T> pt = Parent(t);
                foreach (MLNode<T> temp in pt.Childs)
                {
                    if (temp != null && temp.Data.Equals(t.Data))
                    {
                        return temp;
                    }
                }
                return null;
            }
    
            //把以s为头结点的树插入到树中作为结点t的第i颗子树。成功返回true
            public bool Insert(MLNode<T> s, MLNode<T> t, int i)
            {
                if (IsEmpty()) head = t;
                t = FindNode(t);
                if (t != null && t.Childs.Length > i)
                {
                    t.Childs[i] = s;
                    return true;
                }
                else
                {
                    return false;
                }
            }
    
            //删除结点t的第i个子树。返回第i颗子树的根结点。
            public MLNode<T> Delete(MLNode<T> t, int i)
            {
                t = FindNode(t);
                MLNode<T> node = null;
                if (t != null && t.Childs.Length > i)
                {
                    node = t.Childs[i];
                    t.Childs[i] = null;
                }
                return node;
            }
    
    
            //先序遍历
            //根结点->遍历根结点的左子树->遍历根结点的右子树 
            public void preorder(MLNode<T> root)
            {
                if (root == null)
                    return;
                Console.WriteLine(root.Data + " ");
                for (int i = 0; i < root.Childs.Length; i++)
                {
                    preorder(root.Childs[i]);
                }
            }
    
    
            //后序遍历
            //遍历根结点的左子树->遍历根结点的右子树->根结点
            public void postorder(MLNode<T> root)
            {
                if (root == null)
                { return; }
                for (int i = 0; i < root.Childs.Length; i++)
                {
                    postorder(root.Childs[i]);
                }
                Console.WriteLine(root.Data + " ");
            }
    
    
            //层次遍历
            //引入队列 
            public void LevelOrder(MLNode<T> root)
            {
                Console.WriteLine("遍历开始:");
                if (root == null)
                {
                    Console.WriteLine("没有结点!");
                    return;
                }
    
                MLNode<T> temp = root;
    
                CSeqQueue<MLNode<T>> queue = new CSeqQueue<MLNode<T>>(50);
                queue.EnQueue(temp);
                while (!queue.IsEmpty())
                {
                    temp = (MLNode<T>)queue.DeQueue();
                    Console.WriteLine(temp.Data + " ");
                    for (int i = 0; i < temp.Childs.Length; i++)
                    {
                        if (temp.Childs[i] != null)
                        {
                            queue.EnQueue(temp.Childs[i]);
                        }
                    }
                }
                Console.WriteLine("遍历结束!");
            }
    
            //按某种方式遍历树
            //0 先序
            //1 后序
            //2 层序
            public void Traverse(int TraverseType)
            {
                if (TraverseType == 0) preorder(head);
                else if (TraverseType == 1) postorder(head);
                else LevelOrder(head);
            }
        }
        class Tree
        {
    
            static void Main(string[] args)
            {
                MLTree<string> tree = new MLTree<string>();
                char ch;
                do
                {
                    Console.WriteLine("1.添加结点");
                    Console.WriteLine("2.删除结点");
                    Console.WriteLine("3.遍历树");
                    Console.WriteLine("4.添加子结点");
                    Console.WriteLine("5.退出");
                    Console.WriteLine();
                    ch = Convert.ToChar(Console.ReadLine());
                    switch (ch)
                    {
                        case '1':
                            Console.WriteLine("输入父结点:");
                            string str = Convert.ToString(Console.ReadLine());
                            MLNode<string> pt = new MLNode<string>(4);
                            pt.Data = str;
                            Console.WriteLine("输入子结点:");
                            str = Convert.ToString(Console.ReadLine());
                            MLNode<string> ct = new MLNode<string>(4);
                            ct.Data = str;
                            Console.WriteLine("输入插入子结点的位置:");
                            int i = Convert.ToInt32(Console.ReadLine());
                            bool ok = tree.Insert(ct, pt, i);
                            if (ok)
                            {
                                Console.WriteLine("插入{0}成功", ct.Data);
                                Console.WriteLine(tree.GetDepth(pt));
                            }
                            break;
                        case '2':
                            Console.WriteLine("输入要删除的结点:");
                            str = Convert.ToString(Console.ReadLine());
                            pt = new MLNode<string>(4);
                            pt.Data = str;
                            tree.Delete(pt, 0);
                            break;
                        case '3':
                            tree.Traverse(2);
                            break;
                    }
    
                } while (ch != '5');
            }
        }
    }

    树的遍历

    1.先序遍历

     访问数的根结点->按照从左到右的顺序先序遍历树中的每一颗树

    2.后序遍历

    按照从左到右的顺序后序遍历树中的每一颗子树->访问根结点

    3.层序遍历

    按照树的结构从上到下 从左到右的顺序访问树的结点。

    树的存储方式

    双亲表示法

    孩子链表表示法

    多重链表表示法

    双亲孩子表示法

    孩子兄弟表示法

    注:本文整理自《数据结构(C#语言版)》!!!

     

  • 相关阅读:
    鉴权系统的梳理及并发问题的考虑
    分布式事物解决选择
    电商平台客服和商铺的数据表设计( 初稿记录)43
    问题整理
    关于
    问题整理
    问题整理(常问3连)
    问题整理
    强引用、软引用、弱引用、幻象引用的区别
    通过aop获取切面方法请求入参名称和值的json打印
  • 原文地址:https://www.cnblogs.com/YuanSong/p/2661635.html
Copyright © 2011-2022 走看看