2015-05-06 56 views
6

Mam zestaw ciągów znaków i chcę znaleźć wszystkie możliwe kombinacje ciągów i dodać je do listy. Chcę skończyć z listą wszystkich kombinacji łańcuchów, minus pusty zestaw.Kombinatory C# LINQ: Wszystkie kombinacje zestawu bez pustego zestawu

Stworzyłem rozwiązanie, które robi to dokładnie z zagnieżdżoną pętlą for. Jednak Chcę to zrobić bardziej elegancko, najlepiej z LINQ, i nie jestem tak biegły w tym, ponieważ jestem wciąż całkiem nowy.

Rozwiązanie powinno zawierać 2^n - 1 listy kombinacji, gdzie n jest liczbą oryginalnego zestawu. Oto poprawne przykładem tego, co szukam:

set = {a, b, c} 

completedListOfCombinations = 
{ 
    {a}, 
    {b}, 
    {a, b}, 
    {c}, 
    {a, c}, 
    {b, c}, 
    {a, b, c} 
} 

Oto moja robocza, proste, ale brzydki rozwiązanie, że wykonane z pomocą: https://stackoverflow.com/a/3319652/3371287

List<string> myStrings = new List<string> { "a", "b", "c" }; 

var allCombos = new List<List<string>>(); 

for (int i = 0; i < myStrings.Count; i++) 
{ 
    int subsetCount = allCombos.Count; 
    var m = new List<string>(); 
    m.Add(myStrings[i]); 
    allCombos.Add(m); 

    for (int j = 0; j < subsetCount; j++) 
    { 
     string[] subset = new string[allCombos.ElementAt(j).Count + 1]; 
     allCombos[j].CopyTo(subset, 0); 
     subset[subset.Length - 1] = myStrings[i]; 
     allCombos.Add(subset.ToList()); 
    } 

} 

Może ktoś mi pokazać bardziej elegancki rozwiązanie tego? Widziałem podobne rozwiązania LINQ, które tworzą pary kartezjańskie i listy z progiem, ale nie byłem w stanie dostosować ich do tego, czego potrzebuję.

+0

może to być więcej odpowiednie powyżej [Przegląd kodu] (http://codereview.stackexchange.com) –

+0

możliwy duplikat [Wszystkie kombinacje przedmiotów w tablicach N w C#] (http://stackoverflow.com/questions/13616545/all-combinations-of-items -in-n-array-in-c-sharp) – mbeckish

Odpowiedz

11

Zapewnienie, że wszystkie wartości w listwyjątkowy:

List <String> list = new List<String> { "a", "b", "c" }; 

    var result = Enumerable 
    .Range(1, (1 << list.Count) - 1) 
    .Select(index => list.Where((item, idx) => ((1 << idx) & index) != 0).ToList()); 

Aby wydrukować:

Console.WriteLine(String 
    .Join(Environment.NewLine, result 
    .Select(line => String.Join(", ", line)))); 

Wynik jest

a 
b 
a, b 
c 
a, c 
b, c 
a, b, c 
+0

Tak, wartości na liście są zawsze unikalne. Dobra robota! Chcę to przeanalizować i zrozumieć. – user3371287

+4

Możesz myśleć o tym, licząc od 1 do 2^n-1. Jeśli spojrzysz na indeks binarny (001 -> 010 -> 011 -> 100 -> 101 -> 110 -> 111), wtedy odwzoruj każdą cyfrę binarną na jedną z wartości na liście ("a", "b", " c "), i pokaż tylko te, które są 1 (pokażę 0 jako 0 zamiast ich wcale nie wyświetlać), wtedy otrzymujesz (00a, 0b0, 0ba, c00, c0a, cb0, cba) –