从序列字符串列表创建层次结构
Creating a hierarchical structure from a list of sequence strings
给定一个序列字符串列表,从中创建层次结构的最佳方法是什么?
序列字符串列表类型的示例如下:
List<string> sequenceList = new List<string>() { "1", "1.1", "1.2", "2", "2.1", "2.1.1", "2.1.2" };
我也包含了下面代码的初始传递:
public class Hierarchy
{
public int ID { get; set; }
public int ParentID { get; set; }
}
public IList<Hierarchy> GetHierarchy(IList<string> sequenceList)
{
int iD = 0;
List<Hierarchy> hierarchy = new List<Hierarchy>();
foreach (string sequence in sequenceList)
{
iD++;
List<string> childSequence = new List<string>();
string[] sequenceParts = sequence.Split(new char[] { '.' }, StringSplitOptions.RemoveEmptyEntries);
// If the sequence contains sub-sequence, i.e. "2.1" is a sub-sequence of "2".
if (sequenceParts.Count() > 1)
{
// Struggling with this part, how to obtain the child sequence...
childSequence = sequenceList.Where(s => s.Substring(0, (s.Length - 2)) == sequence.Substring(0, (s.Length - 2))).ToList();
if (childSequence.Count() > 0)
{
int parentID = iD;
foreach (string subSequence in childSequence)
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = parentID });
}
}
else
// Add top level.
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = 0 });
}
return (IList<Hierarchy>)hierarchy;
}
我怀疑我错过了递归的技巧,因此非常感谢任何帮助。
将序列 属性 添加到 class 并使用它来查找父项会有所帮助:
public class Hierarchy
{
public int ID { get; set; }
public int ParentID { get; set; }
public string Sequence { get; set; }
}
public static IList<Hierarchy> GetHierarchy(IList<string> sequenceList)
{
int iD = 0;
List<Hierarchy> hierarchy = new List<Hierarchy>();
foreach (string sequence in sequenceList)
{
iD++;
List<string> childSequence = new List<string>();
string[] sequenceParts = sequence.Split(new char[] { '.' }, StringSplitOptions.RemoveEmptyEntries);
// If the sequence contains sub-sequence, i.e. "2.1" is a sub-sequence of "2".
if (sequenceParts.Count() > 1)
{
var parentSequence = sequence.Substring(0, sequence.LastIndexOf("."));
var parent = hierarchy.Single(x => x.Sequence == parentSequence);
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = parent.ID, Sequence = sequence });
}
else
// Add top level.
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = 0, Sequence = sequence });
}
return (IList<Hierarchy>)hierarchy;
}
已编辑:此版本使用递归
public class Hierarchy
{
public int ID { get; set; }
public int ParentID { get; set; }
}
public static IList<Hierarchy> GetHierarchy(IList<string> sequenceList)
{
int iD = 0;
List<Hierarchy> hierarchy = new List<Hierarchy>();
foreach (string sequence in sequenceList.Where(x => x.Count(f => f == '.') == 0)) // get the root nodes
{
iD++;
var item = new Hierarchy() { ID = iD, ParentID = 0 };
hierarchy.Add(item);
hierarchy.AddRange(GetChildsRecursive(sequence, item.ID, sequenceList, () => ++iD));
}
return (IList<Hierarchy>)hierarchy;
}
private static IList<Hierarchy> GetChildsRecursive(string parentSequence, int parentId, IList<string> sequences, Func<int> idGenerator)
{
var parentDots = parentSequence.Count(f => f == '.');
var childSequences = sequences.Where(x => x.StartsWith(parentSequence) && x.Count(f => f == '.') == parentDots + 1);
var list = new List<Hierarchy>();
foreach (var childSequence in childSequences)
{
var item = new Hierarchy() { ID = idGenerator(), ParentID = parentId };
list.Add(item);
list.AddRange(GetChildsRecursive(childSequence, item.ID, sequences, idGenerator));
}
return list;
}
给定一个序列字符串列表,从中创建层次结构的最佳方法是什么?
序列字符串列表类型的示例如下:
List<string> sequenceList = new List<string>() { "1", "1.1", "1.2", "2", "2.1", "2.1.1", "2.1.2" };
我也包含了下面代码的初始传递:
public class Hierarchy
{
public int ID { get; set; }
public int ParentID { get; set; }
}
public IList<Hierarchy> GetHierarchy(IList<string> sequenceList)
{
int iD = 0;
List<Hierarchy> hierarchy = new List<Hierarchy>();
foreach (string sequence in sequenceList)
{
iD++;
List<string> childSequence = new List<string>();
string[] sequenceParts = sequence.Split(new char[] { '.' }, StringSplitOptions.RemoveEmptyEntries);
// If the sequence contains sub-sequence, i.e. "2.1" is a sub-sequence of "2".
if (sequenceParts.Count() > 1)
{
// Struggling with this part, how to obtain the child sequence...
childSequence = sequenceList.Where(s => s.Substring(0, (s.Length - 2)) == sequence.Substring(0, (s.Length - 2))).ToList();
if (childSequence.Count() > 0)
{
int parentID = iD;
foreach (string subSequence in childSequence)
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = parentID });
}
}
else
// Add top level.
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = 0 });
}
return (IList<Hierarchy>)hierarchy;
}
我怀疑我错过了递归的技巧,因此非常感谢任何帮助。
将序列 属性 添加到 class 并使用它来查找父项会有所帮助:
public class Hierarchy
{
public int ID { get; set; }
public int ParentID { get; set; }
public string Sequence { get; set; }
}
public static IList<Hierarchy> GetHierarchy(IList<string> sequenceList)
{
int iD = 0;
List<Hierarchy> hierarchy = new List<Hierarchy>();
foreach (string sequence in sequenceList)
{
iD++;
List<string> childSequence = new List<string>();
string[] sequenceParts = sequence.Split(new char[] { '.' }, StringSplitOptions.RemoveEmptyEntries);
// If the sequence contains sub-sequence, i.e. "2.1" is a sub-sequence of "2".
if (sequenceParts.Count() > 1)
{
var parentSequence = sequence.Substring(0, sequence.LastIndexOf("."));
var parent = hierarchy.Single(x => x.Sequence == parentSequence);
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = parent.ID, Sequence = sequence });
}
else
// Add top level.
hierarchy.Add(new Hierarchy() { ID = iD, ParentID = 0, Sequence = sequence });
}
return (IList<Hierarchy>)hierarchy;
}
已编辑:此版本使用递归
public class Hierarchy
{
public int ID { get; set; }
public int ParentID { get; set; }
}
public static IList<Hierarchy> GetHierarchy(IList<string> sequenceList)
{
int iD = 0;
List<Hierarchy> hierarchy = new List<Hierarchy>();
foreach (string sequence in sequenceList.Where(x => x.Count(f => f == '.') == 0)) // get the root nodes
{
iD++;
var item = new Hierarchy() { ID = iD, ParentID = 0 };
hierarchy.Add(item);
hierarchy.AddRange(GetChildsRecursive(sequence, item.ID, sequenceList, () => ++iD));
}
return (IList<Hierarchy>)hierarchy;
}
private static IList<Hierarchy> GetChildsRecursive(string parentSequence, int parentId, IList<string> sequences, Func<int> idGenerator)
{
var parentDots = parentSequence.Count(f => f == '.');
var childSequences = sequences.Where(x => x.StartsWith(parentSequence) && x.Count(f => f == '.') == parentDots + 1);
var list = new List<Hierarchy>();
foreach (var childSequence in childSequences)
{
var item = new Hierarchy() { ID = idGenerator(), ParentID = parentId };
list.Add(item);
list.AddRange(GetChildsRecursive(childSequence, item.ID, sequences, idGenerator));
}
return list;
}