13

我有两个排序列表如下:

var list1 = new List<int>() { 1, 1, 1, 2, 3 };
var list2 = new List<int>() { 1, 1, 2, 2, 4 };

我希望输出为:{1, 1, 2}

如何在 C# 中做到这一点?有没有使用 Linq 的方法?

4

5 回答 5

46

使用Intersect

 var commonElements = list1.Intersect(list2).ToList();
于 2012-10-09T15:10:49.623 回答
5

额外的 1 表示您不能使用Intersect,因为它返回一个集合。

这是一些可以满足您需要的代码:

var list1 = new List<int>() { 1, 1, 1, 2, 3 };
var list2 = new List<int>() { 1, 1, 2, 2, 4 };

var grouped1 =
    from n in list1
    group n by n
    into g
    select new {g.Key, Count = g.Count()};

var grouped2 =
    from n in list2
    group n by n
    into g
    select new {g.Key, Count = g.Count()};

var joined =
    from b in grouped2
    join a in grouped1 on b.Key equals a.Key
    select new {b.Key, Count = Math.Min(b.Count, a.Count)};

var result = joined.SelectMany(a => Enumerable.Repeat(a.Key, a.Count));

CollectionAssert.AreEquivalent(new[] {1, 1, 2}, result);
于 2012-10-09T20:39:20.277 回答
3

这很好用:

var list1 = new List<int>() { 1, 1, 1, 2, 3 };
var list2 = new List<int>() { 1, 1, 2, 2, 4 };

var lookup1 = list1.ToLookup(x => x);
var lookup2 = list2.ToLookup(x => x);

var results = lookup1.SelectMany(l1s => lookup2[l1s.Key].Zip(l1s, (l2, l1) => l1));
于 2016-10-09T13:21:13.343 回答
0

虽然@Austin Salonen 的解决方案和@Enigmativity 的解决方案都适用于任何给定的列表,但都没有利用 OP 的条件,即列表是sorted

鉴于这两个列表都是有序的,我们可以O(n + m)及时进行搜索,其中 n 和 m 是每个列表的长度。不完全确定以前的解决方案的性能如何,但肯定会更慢O(n + m)

基本上我们只是遍历两个列表,根据比较检查移动一个或两个枚举数。

var results = new List<int>();
var e1 = list1.GetEnumerator();
var e2 = list2.GetEnumerator();
var hasNext = e1.MoveNext() && e2.MoveNext();

while (hasNext) {
    var value1 = e1.Current;
    var value2 = e2.Current;

    if (value1 == value2) {
        results.Add(value1);
        hasNext = e1.MoveNext() && e2.MoveNext();
    } else if (value1 < value2) {
        hasNext = e1.MoveNext();
    } else if (value1 > value2) {
        hasNext = e2.MoveNext();
    }
}

而已!results如果没有找到匹配项,将是一个空列表。请注意,这假设两个列表都按升序排列。如果它是下降的,只需翻转<and>运算符。

于 2019-02-18T16:44:07.370 回答
-2

我回答这个问题迟了,这可能会对未来的访客有所帮助。

            List<int> p = new List<int> { 1, 1, 1, 2, 3 };
            List<int> q = new List<int> { 1, 1, 2, 2, 4 };
            List<int> x = new List<int>();
            for (int i = 0; i < p.Count; i++ )
            {
                if (p[i] == q[i])
                {
                    x.Add(p[i]);
                }
            }
于 2016-10-09T13:12:40.670 回答