C # 6 добавляет новую функцию только для этого: расширение Добавить методы. Это всегда было возможно для VB.net, но теперь доступно в C #.
Теперь вам не нужно напрямую добавлять Add()
методы в ваши классы, вы можете реализовать их как методы расширения. Расширяя любой перечислимый тип методом Add()
, вы сможете использовать его в выражениях инициализатора коллекции. Поэтому вам больше не нужно явно получать списки (, как указано в другом ответе ), вы можете просто расширить его.
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...
}
Это позволит вам сделать это для любого класса, который реализует 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 },
};
Конечно, вы не ограничены расширением коллекций кортежей, это может быть для коллекций любого определенного типа, для которого вы хотите специальный синтаксис.
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 будет добавлять поддержку кортежей, встроенных в язык, хотя они будут другого типа (вместо System.ValueTuple
). Поэтому было бы хорошо добавить перегрузки для кортежей значений, чтобы у вас была возможность использовать их. К сожалению, между ними нет неявных преобразований.
public static class ValueTupleListExtensions
{
public static void Add<T1, T2>(this IList<Tuple<T1, T2>> list,
ValueTuple<T1, T2> item) => list.Add(item.ToTuple());
}
Таким образом, инициализация списка будет выглядеть еще лучше.
var points = new List<Tuple<int, int, int>>
{
(0, 0, 0),
(1, 2, 3),
(-1, 12, -73),
};
Но вместо того, чтобы преодолевать все эти проблемы, возможно, было бы лучше перейти исключительно на использование ValueTuple
.
var points = new List<(int, int, int)>
{
(0, 0, 0),
(1, 2, 3),
(-1, 12, -73),
};