C # получить отличные значения из неровного массива - PullRequest
2 голосов
/ 30 января 2012

Я ищу способ получить отличные значения из зубчатого массива. Я пытался поместить его в словарь, но он не видит значения как отдельные. У кого-нибудь есть идеи для решения? Вот что у меня есть:

int[][] connectionList = GetArray();

Пример данных, возвращаемых из этого:

[0][1, 130, 136, 138]
[1][1, 78, 126, 138]
[2][1, 10, 125, 138]
[3][1, 130, 136, 138]
[4][1, 78, 126, 138]
[5][1, 130, 136, 138]
[6][1, 72, 135, 138]
[7][1, 73, 135, 138]
[8][1, 130, 136, 138]

Попробовал добавить в словарь. Я пытался поймать повторяющиеся значения по мере их добавления, но это не сработало, поэтому попытался добавить .Distinct (), но там тоже нет радости

Dictionary<int, int[]> myDictionary = new Dictionary<int, int[]>();
for (int i = 0; i < connectionList.Length; i++)
{ 
    List<int> list = new List<int>();
    for (int j = 0; j < connectionList[i].Length; j++)
    {
        list.Add(connectionList[i][j]);        
    }
    if (myDictionary.Where(x => x.Value == list.ToArray()).Count() == 0)
        myDictionary.Add(i, list.ToArray());
}
var distinctList = myDictionary.Values.Distinct().ToList();

Из приведенного выше списка вывод, который я ищу, будет:

[0][1, 130, 136, 138]
[1][1, 78, 126, 138]
[2][1, 10, 125, 138]
[4][1, 72, 135, 138]
[5][1, 73, 135, 138]

Есть мысли, как мне этого достичь?

Ответы [ 3 ]

4 голосов
/ 30 января 2012

Вот один из способов сделать это:

var distinctList = connectionList.GroupBy(x => string.Join(",", x))
                                 .Select(g => g.First())
                                 .ToList();

Хотя, вероятно, предпочтительнее создать пользовательский IEqualityComparer<T> - как предложено Crab Bucket и Тобиас - вместо создания одноразовой строки для сравнения.

2 голосов
/ 30 января 2012

Это будет легко с LINQ Distinct, все, что вам нужно, это предоставить собственную реализацию для IEqualityComparer:

public class IntArrayComparer : IEqualityComparer<int[]>
{
    public bool Equals(int[] i1, int[] i2)
    {
        if(ReferenceEquals(i1, i2))
        {
            return true;
        }
        else if(i1 == null || i2 == null)
        {
            return false;
        }
        else if(i1.Length != i2.Length)
        {
            return false;
        }

        for(int i = 0; i < i1.Length; ++i)
        {
            if(i1[i] != i2[i]) return false;
        }

        return true;
    }

    public int GetHashCode(int[] obj)
    {
        // Average is probably not the best hash for an int array,
        // but I'm lazy right now and this is only for demonstration purposes
        return obj != null ? (int)obj.Average() : 0;
    }
}

И используйте это в своем коде, например так:

int[][] connectionList = GetArray().Distinct(new IntArrayComparer()).ToArray();
1 голос
/ 30 января 2012

Не могли бы вы использовать IEqualityComparer

 public class MyComparer : IEqualityComparer<int []> 
    {     
        bool IEqualityComparer<int[]>.Equals(int[] x, int[] y)     
        {         
             //.. your particular comparison logic goes here
        }
        int IEqualityComparer<int[]>.GetHashCode(int [] obj)     
        {         
            return obj.GetHashCode();     
        }     

    } 

Тогда звоните вот так

var distinctList = myDictionary.Values.Distinct(new MyComparer()).ToList(); 
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...