C # 6 dodaje nową funkcję tylko do tego: rozszerzenie Dodaj metody. Zawsze było to możliwe dla VB.net, ale jest teraz dostępne w języku C #.
Teraz nie musisz dodawać Add()
metod bezpośrednio do swoich klas, możesz je zaimplementować jako metody rozszerzeń. Rozszerzając dowolny typ wyliczalny za pomocą Add()
metody, będziesz mógł używać go w wyrażeniach inicjalizujących kolekcję. Więc nie musisz już czerpać z list wyraźnie ( jak wspomniano w innej odpowiedzi ), możesz po prostu go rozszerzyć.
public static class TupleListExtensions
{
public static void Add<T1, T2>(this IList<Tuple<T1, T2>> list,
T1 item1, T2 item2)
{
list.Add(Tuple.Create(item1, item2));
}
public static void Add<T1, T2, T3>(this IList<Tuple<T1, T2, T3>> list,
T1 item1, T2 item2, T3 item3)
{
list.Add(Tuple.Create(item1, item2, item3));
}
// and so on...
}
To pozwoli ci to zrobić na dowolnej klasie, która implementuje IList<>
:
var numbers = new List<Tuple<int, string>>
{
{ 1, "one" },
{ 2, "two" },
{ 3, "three" },
{ 4, "four" },
{ 5, "five" },
};
var points = new ObservableCollection<Tuple<double, double, double>>
{
{ 0, 0, 0 },
{ 1, 2, 3 },
{ -4, -2, 42 },
};
Oczywiście nie jesteś ograniczony do rozszerzania kolekcji krotek, może to być dla zbiorów dowolnego określonego typu, dla którego chcesz specjalnej składni.
public static class BigIntegerListExtensions
{
public static void Add(this IList<BigInteger> list,
params byte[] value)
{
list.Add(new BigInteger(value));
}
public static void Add(this IList<BigInteger> list,
string value)
{
list.Add(BigInteger.Parse(value));
}
}
var bigNumbers = new List<BigInteger>
{
new BigInteger(1), // constructor BigInteger(int)
2222222222L, // implicit operator BigInteger(long)
3333333333UL, // implicit operator BigInteger(ulong)
{ 4, 4, 4, 4, 4, 4, 4, 4 }, // extension Add(byte[])
"55555555555555555555555555555555555555", // extension Add(string)
};
C # 7 będzie dodawać obsługę krotek wbudowanych w język, choć będą one innego typu ( System.ValueTuple
zamiast tego). Dlatego dobrze byłoby dodać przeciążenia dla krotek wartości, abyś miał również możliwość ich użycia. Niestety nie ma między nimi żadnych domyślnych konwersji.
public static class ValueTupleListExtensions
{
public static void Add<T1, T2>(this IList<Tuple<T1, T2>> list,
ValueTuple<T1, T2> item) => list.Add(item.ToTuple());
}
W ten sposób inicjalizacja listy będzie wyglądać jeszcze ładniej.
var points = new List<Tuple<int, int, int>>
{
(0, 0, 0),
(1, 2, 3),
(-1, 12, -73),
};
Ale zamiast przejść przez te wszystkie kłopoty, może być po prostu lepiej przejść na używanie ValueTuple
wyłącznie.
var points = new List<(int, int, int)>
{
(0, 0, 0),
(1, 2, 3),
(-1, 12, -73),
};