【问题标题】:Implementing IEnumerable on a tree structure在树结构上实现 IEnumerable
【发布时间】:2010-12-28 01:41:27
【问题描述】:

基于这些人的工作:

我正在尝试实现一个可以这样使用的 TreeView 助手:

<%= Html.TreeView("records", 
                  Library.Instance.Records, 
                  r => r.Children, 
                  r => r.ID) %>

而树结构是这样定义的:

public class Tree<T> : TreeNode<T> where T : TreeNode<T>
{ }


public class TreeNode<T> : IDisposable where T : TreeNode<T>
{
    public T Parent { get; set; }
    public TreeNodeList<T> Children { get; set; }
}


public class TreeNodeList<T> : List<TreeNode<T>> where T : TreeNode<T>
{
    public T Parent;

    public T Add(T node)
    {
        base.Add(node);
        node.Parent = (T)Parent;
        return node;
    }

    public void Remove(T node)
    {
        if (node != null)
            node.Parent = null;
        base.Remove(node);
    }
}

TreeView 助手有这个签名:

public static string TreeView<T>(this HtmlHelper htmlHelper, string treeId,
   IEnumerable<T> rootItems, Func<T, IEnumerable<T>> childrenProperty, 
   Func<T, string> itemContent, bool includeJavascript, string emptyContent)
{
    ...
}   

因此,我需要我的 Tree 结构来实现 IEnumerable,这样我就可以将它与 TreeView 助手一起使用,这就引出了一个问题:在这种情况下,我将在哪里以及如何实现 IEnumerable?

【问题讨论】:

    标签: c# collections tree ienumerable treenode


    【解决方案1】:

    我不完全了解您的树结构的确切细节,但这里有一个简单的实现,它采用一个通用的节点树并将其递归地呈现到 html 列表中。

    public static string TreeView<T>(IEnumerable<T> rootItems,
                                     Func<T, IEnumerable<T>> childrenProperty,
                                     Func<T, string> itemContent)
    {
        if (rootItems == null || !rootItems.Any()) return null;
    
        var builder = new StringBuilder();
        builder.AppendLine("<ul>");
    
        foreach (var item in rootItems)
        {
            builder.Append("  <li>").Append(itemContent(item)).AppendLine("</li>");
            var childContent = htmlHelper.TreeView(treeId,
                                                   childrenProperty(item),
                                                   childrenProperty,
                                                   itemContent);
    
            if (childContent != null)
            {
                var indented = childContent.Replace(Environment.NewLine,
                                                    Environment.NewLine + "  ");
                builder.Append("  ").AppendLine(indented);
            }
        }
    
        builder.Append("</ul>");
        return builder.ToString();
    }
    

    我使用的节点类比较简单,只有两个属性。

    public class Node<T>
    {
        public Node(T data)
        {
            Data = data;
            Children = new List<Node<T>>();
        }
    
        public T Data { get; private set; }
        public ICollection<Node<T>> Children { get; private set; }
    }
    

    这是一些将树输出到控制台的测试代码。

    var Records = new[] {
        new Node<string>("one") {
            Children = {
                new Node<string>("one-one") {
                    Children = {
                        new Node<string>("one-one-one"),
                        new Node<string>("one-one-two"),
                        new Node<string>("one-one-three")
                    }
                },
                new Node<string>("one-two"),
                new Node<string>("one-three")
            }
        },
        new Node<string>("two"),
        new Node<string>("three")
    };
    Console.WriteLine(TreeView(Records,
                               r => r.Children,
                               r => r.Data));
    

    这是上面代码的结果。

    <ul>
      <li>one</li>
      <ul>
        <li>one-one</li>
        <ul>
          <li>one-one-one</li>
          <li>one-one-two</li>
          <li>one-one-three</li>
        </ul>
        <li>one-two</li>
        <li>one-three</li>
      </ul>
      <li>two</li>
      <li>three</li>
    </ul>
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-02-10
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多