Quel est l'équivalent C # des opérations sur les tranches Python?
my_list = ['a', 'b', 'c', 'd', 'e', 'f', 'g']
result1 = my_list[2:4]
result2 = my_list[1:]
result3 = my_list[:3]
result4 = my_list[:3] + my_list[4:]
Une partie est couverte ici , mais c'est moche et ne traite pas de toutes les utilisations du découpage en tranches au point de ne pas répondre évidemment à la question.
Si vous avez une List
GetRange peut être utile.
À partir du lien MSDN:
Une copie superficielle d'une collection de types de référence, ou d'un sous-ensemble de celui-ci collection, contient uniquement les références aux éléments du fichier collection. Les objets eux-mêmes ne sont pas copiés. Les références dans la nouvelle liste pointe vers les mêmes objets que les références dans le fichier liste originale.
La fonction Slice
peut alors être:
public static IEnumerable<T> Slice<T>(this List<T> source, int from, int to) => source.GetRange(from, to - from);
Les plages négatives prises en charge par les tranches python peuvent également être gérées avec une perte de propreté.
À partir de C#8
, le découpage en tranches devient beaucoup plus facile pour les structures de données indexées.
var result1 = myList[2..5]; // end (5) is exclusive
var result2 = myList[1..^0]; // from index 1 to the end
var result3 = myList[0..3]; // end (3) exclusive
De cette façon, vous n'avez pas à soustraire
public static IEnumerable<A> Slice<A> (int from, int to, IEnumerable<A> e) {
return e.Take (to).Skip (from);
}
public static T[] slice<T>(T[] l, int from, int to)
{
T[] r = new T[to - from];
for (int i = from; i < to; i++)
{
r[i-from]=l[i];
}
return r;
}
Ecrire une extension personnalisée:
public static List<T> Slice<T>(this List<T> li, int start, int end)
{
if (start < 0) // support negative indexing
{
start = li.Count + start;
}
if (end < 0) // support negative indexing
{
end = li.Count + end;
}
if (start > li.Count) // if the start value is too high
{
start = li.Count;
}
if (end > li.Count) // if the end value is too high
{
end = li.Count;
}
var count = end - start; // calculate count (number of elements)
return li.GetRange(start, count); // return a shallow copy of li of count elements
}
Quelques tests:
[Fact]
public void Slice_list()
{
var li1 = new List<char> {'a', 'b', 'c', 'd', 'e', 'f', 'g'};
Assert.Equal(new List<char> {'c', 'd'}, li1.Slice(2, 4));
Assert.Equal(new List<char> {'b', 'c', 'd', 'e', 'f', 'g'}, li1.Slice(1, li1.Count));
Assert.Equal(new List<char> {'a', 'b', 'c'}, li1.Slice(0, 3));
Assert.Equal(li1, li1.Slice(0, 4).Concat(li1.Slice(4, li1.Count)));
Assert.Equal(li1, li1.Slice(0, 100));
Assert.Equal(new List<char>(), li1.Slice(100, 200));
Assert.Equal(new List<char> {'g'}, li1.Slice(-1, li1.Count));
Assert.Equal(new List<char> {'f', 'g'}, li1.Slice(-2, li1.Count));
Assert.Equal(new List<char> {'a', 'b', 'c', 'd', 'e', 'f'}, li1.Slice(0, -1));
Assert.Equal(new List<char> {'c', 'd', 'e'}, li1.Slice(2, -2));
}