개체 목록에서 고유 한 필드 값을 반환하는 linq 쿼리


88
class obj
{
    int typeId; //10 types  0-9 
    string uniqueString; //this is unique
}

100 개의 obj 요소가있는 목록이 있지만 고유 한 typeID는 10 개 뿐이라고 가정합니다.
LINQ 쿼리를 작성하여 objs 목록에서 10 개의 고유 한 정수를 반환 할 수 있습니까?


이 질문은 너무 쉽고 현상금에 적합하지 않습니다.
Thinker

답변:


155
objList.Select(o=>o.typeId).Distinct()

2
두 번째 형식은 내가 아는 한 존재하지 않는 Distinct 과부하를 사용하는 것 같습니다.
Jon Skeet

죄송합니다. @Jon Skeet에게 감사합니다. IEqualityComparer로 수행 할 수있는 두 번째 항목을 제거했습니다.
Arsen Mkrtchyan 2011 년

3
Jon Skeet이 언급했듯이 이것은 Select에 지정된 속성 만 반환합니다.
Peet vd Westhuizen

58

전체 개체를 원하지만으로 구별 만 처리하고 싶다고 가정하면 typeIDLINQ에이 작업을 쉽게 수행 할 수있는 것은 없습니다. (이 경우 단지 합니다 typeID-와 그에게 프로젝트 값을, 그것은 쉽게 Select하고 정상적인 사용 Distinct전화를.)

에서 MoreLINQ 우리는이 DistinctBy당신이 사용할 수있는 연산자를 :

var distinct = list.DistinctBy(x => x.typeID);

이것은 LINQ to Objects에서만 작동합니다.

그룹화 또는 조회를 사용할 수 있지만 다소 성 가시고 비효율적입니다.

var distinct = list.GroupBy(x => x.typeID, (key, group) => group.First());

DistinctBy을하려면 네임 스페이스의 Microsoft.Ajax.Utilities 추가해야 표시합니다
Shiljo 폴슨

1
@Shil : 아니요, MoreLINQ에서 DistinctBy에 대해 썼습니다. Microsoft.Ajax.Utilities와 관련이 없습니다.
Jon Skeet

이제 LINQ에 IEqualityComparer를 매개 변수로 사용하고 IEqualityComparer의 메서드 구현에 따라 별개의 개체 목록을 반환하는 Distinct 오버로드가 있음을 알 수 있습니다.
Dipendu Paul

1
@DipenduPaul : 예,하지만 여전히 주어진 속성에 대해 동등 비교자를 만드는 것을 의미합니다. 이는 성 가시고 읽기 어렵게 만듭니다. MoreLINQ 종속성을 취할 수 있다면 더 깨끗하다고 ​​생각합니다.
Jon Skeet

22

순수한 Linq를 사용하려면 groupby를 사용할 수 있습니다.

List<obj> distinct =
  objs.GroupBy(car => car.typeID).Select(g => g.First()).ToList();

MoreLinq 가 수행 하는 것과 유사한 방법을 앱 전체에서 사용하려는 경우 :

public static IEnumerable<TSource> DistinctBy<TSource, TKey>
    (this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
{
    HashSet<TKey> seenKeys = new HashSet<TKey>();
    foreach (TSource element in source)
    {
        if (!seenKeys.Contains(keySelector(element)))
        {
            seenKeys.Add(keySelector(element));
            yield return element;
        }
    }
}

이 방법을 사용하여 Id 속성 만 사용하여 고유 한 값을 찾으려면 다음을 사용할 수 있습니다.

var query = objs.DistinctBy(p => p.TypeId);

여러 속성을 사용할 수 있습니다.

var query = objs.DistinctBy(p => new { p.TypeId, p.Name });

여러 속성 부분에 감사드립니다 !
Nae

7

물론 Enumerable.Distinct입니다..

obj(예 :) 모음이 주어지면 foo다음과 같이 할 수 있습니다.

var distinctTypeIDs = foo.Select(x => x.typeID).Distinct();

5

나는 이것이 당신이 찾고있는 것이라고 생각합니다.

    var objs= (from c in List_Objects 
orderby c.TypeID  select c).GroupBy(g=>g.TypeID).Select(x=>x.FirstOrDefault());      

LINQ를 사용하여 고유 한 IQueryable 반환과 유사 합니까?


.First그 안에 뭔가가 없다면 그룹이 없을 것이기 때문에 괜찮습니다.
mqp 2011-06-03

1
대체 오버로드를 사용하여 GroupBy이것을 더 간단하게 만들 수도 있습니다 . 예제는 내 대답을 참조하십시오.
Jon Skeet 2011 년

3

Linq를 사용하려는 경우 EqualsGetHashCode 메서드를 재정의 할 수 있습니다 .

제품 등급 :

public class Product
{
    public string ProductName { get; set; }
    public int Id { get; set; }


    public override bool Equals(object obj)
    {
        if (!(obj is Product))
        {
            return false;
        }

        var other = (Product)obj;
        return Id == other.Id;
    }

    public override int GetHashCode()
    {
        return Id.GetHashCode();
    }
}

주요 방법 :

static void Main(string[] args)
    {

        var products = new List<Product>
        {
            new Product{ ProductName="Product 1",Id = 1},
            new Product{ ProductName="Product 2",Id = 2},
            new Product{ ProductName="Product 4",Id = 5},
            new Product{ ProductName="Product 3",Id = 3},
            new Product{ ProductName="Product 4",Id = 4},
            new Product{ ProductName="Product 6",Id = 4},
            new Product{ ProductName="Product 6",Id = 4},
        };

        var itemsDistinctByProductName = products.Distinct().ToList();

        foreach (var product in itemsDistinctByProductName)
        {
            Console.WriteLine($"Product Id : {product.Id} ProductName : {product.ProductName} ");
        }

        Console.ReadKey();
    }

1

특정 데이터를 드롭 다운에 바인딩하고 싶었고 고유해야합니다. 다음을 수행했습니다.

List<ClassDetails> classDetails;
List<string> classDetailsData = classDetails.Select(dt => dt.Data).Distinct.ToList();
ddlData.DataSource = classDetailsData;
ddlData.Databind();

도움이되는지 확인

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.