2 List <T>를 병합하고 C #에서 중복 값을 제거하는 방법


159

세 번째 목록으로 결합하고 해당 목록에서 중복 값을 제거 해야하는 두 개의 목록이 있습니다.

조금 설명하기 어려우므로 코드의 모양과 결과로 원하는 결과의 예를 보여 드리겠습니다. 샘플에서는 ResultAnalysisFileSql 클래스가 아닌 int 유형을 사용합니다.

first_list = [1, 12, 12, 5]

second_list = [12, 5, 7, 9, 1]

두 목록을 결합한 결과는이 목록이됩니다. 결과 _ 목록 = [1, 12, 5, 7, 9]

결과에는 두 개의 "12"값을 포함하여 첫 번째 목록이 있고 second_list에는 추가 12, 1 및 5 값이 있습니다.

ResultAnalysisFileSql 클래스

[Serializable]
    public partial class ResultAnalysisFileSql
    {
        public string FileSql { get; set; }

        public string PathFileSql { get; set; }

        public List<ErrorAnalysisSql> Errors { get; set; }

        public List<WarningAnalysisSql> Warnings{ get; set; }

        public ResultAnalysisFileSql()
        {

        }

        public ResultAnalysisFileSql(string fileSql)
        {
            if (string.IsNullOrEmpty(fileSql)
                || fileSql.Trim().Length == 0)
            {
                throw new ArgumentNullException("fileSql", "fileSql is null");
            }

            if (!fileSql.EndsWith(Utility.ExtensionFicherosErrorYWarning))
            {
                throw new ArgumentOutOfRangeException("fileSql", "Ruta de fichero Sql no tiene extensión " + Utility.ExtensionFicherosErrorYWarning);
            }

            PathFileSql = fileSql;
            FileSql = ObtenerNombreFicheroSql(fileSql);
            Errors = new List<ErrorAnalysisSql>();
            Warnings= new List<WarningAnalysisSql>();
        }

        private string ObtenerNombreFicheroSql(string fileSql)
        {
            var f = Path.GetFileName(fileSql);
            return f.Substring(0, f.IndexOf(Utility.ExtensionFicherosErrorYWarning));
        }


        public override bool Equals(object obj)
        {
            if (obj == null)
                return false;
            if (!(obj is ResultAnalysisFileSql))
                return false;

            var t = obj as ResultAnalysisFileSql;
            return t.FileSql== this.FileSql
                && t.PathFileSql == this.PathFileSql
                && t.Errors.Count == this.Errors.Count
                && t.Warnings.Count == this.Warnings.Count;
        }


    }

중복을 결합하고 제거하기위한 샘플 코드?

답변:


288

Enumerable을 살펴 보셨습니까?

이 메소드는 리턴 세트에서 중복을 제외합니다 . 이것은 Concat 메서드와는 다른 동작으로, 중복을 포함하여 입력 시퀀스의 모든 요소를 ​​반환합니다.

List<int> list1 = new List<int> { 1, 12, 12, 5};
List<int> list2 = new List<int> { 12, 5, 7, 9, 1 };
List<int> ulist = list1.Union(list2).ToList();

// ulist output : 1, 12, 5, 7, 9

6
@ Dr TJ : 당신의 사람 클래스는 IEqualityComparer <T>를 구현합니까? 그렇다면 GetHashCode 및 Equals 메소드를 확인해야합니다. msdn.microsoft.com/en-us/library/bb341731.aspx 의 설명 섹션을 참조 하십시오 .
Tomas Narros

1
나는 2 개 개의 다른 컬렉션에서이 작업을 사용하는 문제에 달려 있기 때문에주의하는 것이 중요 : "당신 수 없습니다 연합 (EU) 두 가지 종류, 다른 하나 상속하지 않는 한"에서 stackoverflow.com/a/6884940/410937 산출 cannot be inferred from the usage오류입니다.
atconway

30

왜 단순히 예를 들어

var newList = list1.Union(list2)/*.Distinct()*//*.ToList()*/;

오 ... 따라가하는 MSDN 당신이 밖으로 떠날 수있는.Distinct()

이 메소드는 리턴 세트에서 중복을 제외합니다.


25

Union의 성능이 좋지 않습니다.이 기사에서는 서로 비교하는 방법에 대해 설명합니다.

var dict = list2.ToDictionary(p => p.Number);
foreach (var person in list1)
{
        dict[person.Number] = person;
}
var merged = dict.Values.ToList();

목록 및 LINQ 병합 : 4820ms
사전 병합 : 16ms
HashSet 및 IEqualityComparer : 20ms
LINQ Union 및 IEqualityComparer : 24ms


1
또한 사전 병합 사용의 또 다른 이점-> DB 데이터에서 두 가지 목록이 다시 나타납니다. 내 데이터에는 타임 스탬프 필드가 있는데 두 데이터 목록에서 다릅니다. 유니온을 사용하면 타임 스탬프가 다르기 때문에 중복됩니다. 그러나 병합을 통해 사전에서 고려할 고유 필드를 결정할 수 있습니다. +1
JimSan

프로세서 속도에 따라 다를 수 있으며 사용중인 CPU 종류에 따라 다릅니다.
Asad Ali

7
그리고이 기사의 끝에 "LINQ Union은 의도를 매우 명확하게 전달하기 때문에 LINQ Union을 선호합니다."라고 말합니다. ;) (도있었습니다 만 8 MS 차이)
제임스 윌킨스

1
차이가 무시할 수있는 작은 목록의 경우 Union더 깨끗하고 읽기 쉬운 코드가 생성됩니다. 속도가 느리지 않을 때 코드를 과도하게 최적화하는 데 시간을 소비하면 유지 보수에 따른 불이익이 발생할 수 있습니다.
elolos 2016 년

14

Linq의 조합 사용 :

using System.Linq;
var l1 = new List<int>() { 1,2,3,4,5 };
var l2 = new List<int>() { 3,5,6,7,8 };
var l3 = l1.Union(l2).ToList();

11
    List<int> first_list = new List<int>() {
        1,
        12,
        12,
        5
    };

    List<int> second_list = new List<int>() {
        12,
        5,
        7,
        9,
        1
    };

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