我有不同颜色的形状。
Shape pink = new Shape() { name = "Pink" };
Shape yellow = new Shape() { name = "Yellow" };
Shape red = new Shape() { name = "Red" };
Shape white = new Shape() { name = "White" };
Shape blue = new Shape() { name = "Blue" };
每个形状都会返回其触摸的其他任何形状的列表,并存储在列表中。
List<List<Shape>> lists;
所以列表看起来像这样
lists = new List<List<Shape>>()
{
new List<Shape>() { pink, yellow },
new List<Shape>() { yellow, pink, red },
new List<Shape>() { red, yellow},
new List<Shape>() { white, blue},
new List<Shape>() { blue, white}
};
我想浓缩一下,并完成为一个新的触摸形状列表列表。
List<List<Shape>> result
在这种情况下,结果仅包含两个
List<Shape>
例如
{{pink, yellow, red}, { white, blue}}
子列表共享某些共同点的地方。
我无法通过循环使用它,而且我对Linq不太熟悉。
另一种情况是
lists = new List<List<Shape>>()
{
new List<Shape>() { pink, yellow },
new List<Shape>() { yellow, pink, red },
new List<Shape>() { red, yellow, blue},
new List<Shape>() { white, blue,},
new List<Shape>() { blue, white, red}
};
结果列表应只包含一个列表
{{pink, yellow, red, blue, white}}
因为所有以前的列表都有一些相对的颜色。
答案 0 :(得分:1)
我也尝试过使用linq。只需将弦替换为您的形状即可。但是字符串使atm更容易获得算法的思想。 请检查代码中的注释以了解不同的步骤:
var lists = new List<List<string>>();
lists.Add(new List<string> { "a", "b", "c" });
lists.Add(new List<string> { "a", "c" });
lists.Add(new List<string> { "d", "e" });
lists.Add(new List<string> { "e", "d" });
lists.Add(new List<string> { "e", "a" }); // from my comment
var results = new List<List<string>>();
foreach (var list in lists)
{
// That checks, if for this list, there is already a list, that contains all the items needed.
if (results.Any(r => r.Count == r.Union(list).Count()))
{
continue;
}
// get the lists, that contains at least one item of the current "list".
// This is important, as depending on the amount of elements, there have to be specific further steps.
var listsWithItemsOfList = results.Where(r => list.Any(x => r.Contains(x)));
// if not item, then you just have to add the whole content, as non of the colors exist.
if (!listsWithItemsOfList.Any())
{
results.Add(new List<string>(list));
}
// if exactly, one, that add all the items, that were missing
// (it might be, that nothing is added in case list.Except(l) is empty.
else if(listsWithItemsOfList.Count() == 1)
{
var listWithOneItem = listsWithItemsOfList.Single();
listWithOneItem.AddRange(list.Except(listWithOneItem));
}
else
{
// if multiple elements, it's getting complicated.
// It means, that all needed items are currently spreaded over multiple lists, that have now to be merged.
var newMergedList = listsWithItemsOfList.SelectMany(x => x).Distinct().ToList(); // merge all into one
results.RemoveAll(x => listsWithItemsOfList.Contains(x)); // remove those lists from results
results.Add(newMergedList); // just add one new list, containing all.
}
}
答案 1 :(得分:0)
这是我的尝试,结合使用linq和loops。 (IME表示有可能完全在linq中完成此操作,这有使其难以阅读的风险)
我首先对具有最长列表的输入进行排序,然后查看是否存在包含输入中每一项的现有输出-如果没有,则添加新项。
var yellow = 0;
var pink = 1;
var red = 2;
var white = 3;
var blue = 4;
var input = new List<List<int>> {
new List<int> { pink, yellow },
new List<int> { yellow, pink, red},
new List<int> { red, yellow},
new List<int> { white, blue},
new List<int> { blue, white}
};
var output = new List<List<int>>();
// Start with the longest lists
foreach (var item in input.OrderByDescending(x => x.Count))
{
// See if it will fit in an existing output value
var itemIsEntirelyContainedByExistingOutput = false;
foreach (var outputValue in output)
{
if (item.All(colour => outputValue.Contains(colour)))
{
itemIsEntirelyContainedByExistingOutput = true;
break;
}
}
// No, so add this to the list of outputs
if (!itemIsEntirelyContainedByExistingOutput)
{
output.Add(item);
}
}
这里是将其压缩为linq的尝试。在这一点上,调试起来要困难得多,尽管我希望它仍然可读。
// Start with the longest lists
foreach (var item in input.OrderByDescending(x => x.Count))
{
// See if it will fit in an existing output value
if (!output.Any(x => item.All(x.Contains)))
{
// No, so add this to the list of outputs
output.Add(item);
}
}
答案 2 :(得分:0)
我想我现在已经明白了这个问题。 输入定义哪些颜色链接到其他颜色,结果是链接颜色的列表。
// Build a list of distinct colours
var allColours = input.SelectMany(x => x.Select(y => y)).Distinct();
foreach (var colour in allColours)
{
// Find all colours linked to this one
var linkedColours = input.Where(x => x.Contains(colour)).SelectMany(x => x.Select(y => y)).Distinct().ToList();
// See if any of these colours are already in the results
var linkedResult = results.FirstOrDefault(x => x.Any(y => linkedColours.Contains(y)));
if (linkedResult == null)
{
// Create a new result
results.Add(linkedColours);
}
else
{
// Add missing colours to the result
linkedResult.AddRange(linkedColours.Where(x => !linkedResult.Contains(x)));
}
}