Перечислитель как аргумент


Хорошо, давайте начнем с этого очень простого метода нажатия кнопки

    private void button1_Click(object sender, EventArgs e)
    {
        int counter = 1;
        List<int> items = new int[] { 1, 2, 3 }.ToList();
        List<int>.Enumerator enm = items.GetEnumerator();

        // 1
        if (!enm.MoveNext())
            throw new Exception("Unexpected end of list");
        if (enm.Current != counter)
            throw new Exception(String.Format("Expect {0} but actual {1}", counter, enm.Current));
        counter++;

        // 2
        if (!enm.MoveNext()) 
            throw new Exception("Unexpected end of list");
        if (enm.Current != counter) 
            throw new Exception(String.Format("Expect {0} but actual {1}", counter, enm.Current));
        counter++;

        //3
        if (!enm.MoveNext())
            throw new Exception("Unexpected end of list");
        if (enm.Current != counter)
            throw new Exception(String.Format("Expect {0} but actual {1}", counter, enm.Current));
        counter++;

        if (enm.MoveNext()) 
            throw new Exception("Unexpected continuation of list");
    }

Этот метод ничего не делает, потому что каждое утверждение изящно проходит. Все в порядке, пока я не считаю, что должен ввести метод удаления избыточности

    static void AssertNext(ref int counter, List<int>.Enumerator e)
    {
        if (!e.MoveNext()) 
            throw new Exception("Unexpected end of list");
        if (e.Current != counter) 
            throw new Exception(String.Format("Expect {0} but actual {1}", counter, e.Current));
        counter++;
    }

    private void button2_Click(object sender, EventArgs e)
    {
        var counter = 1;
        var items = new int[] { 1, 2, 3 }.ToList();
        var enm = items.GetEnumerator();
        AssertNext(ref counter, enm);
        AssertNext(ref counter, enm);
        AssertNext(ref counter, enm);
        if (enm.MoveNext()) throw new Exception("Unexpected continuation of list");
    }
Хотя этот рефакторинг прост (по крайней мере, для меня). Это действительно нарушает программу ! При втором вызове AssertNext создается впечатление, что перечислитель уже был сброшен в начальную точку и вызвал сбой утверждения.

I не могу понять, что происходит. Я действительно чувствую себя новичком с этой головоломкой.

Чего мне здесь не хватает ?

2 4

2 ответа:

Мне кажется, что это как-то связано со списком.Перечислитель - это структура. Вы передаете его методу, манипулируете им, а затем возвращаетесь. Манипуляция, вероятно, не произойдет для вашего первоначального экземпляра.

List<T>.Enumerator это тип значения, означающий, что он копируется в локальную область вашего метода, изменяется, а затем уничтожается при выходе из метода. Попробуйте передать его по ссылке тоже.