Anders
Anders

Reputation: 12560

Is it possible to do a For...Each Loop Backwards?

I don't believe this is possible by conventional methods, but something like this verbose code:

For Each s As String In myStringList Step -1
    //' Do stuff here
Next

I will probably have to invert the myString object before a conventional For..Each Loop, correct?

Upvotes: 40

Views: 84215

Answers (14)

MontanaMan
MontanaMan

Reputation: 212

This works:

Dim myArray() As String = {"1", "2", "3", "4", "5"}

For Each n As String In myArray.Reverse
    Debug.Print(n)
Next

Output: 5 4 3 2 1

'

This also works:

Dim myArray() As String = {"1", "2", "3", "4", "5"}

For i As Integer = myArray.Length - 1 To 0 Step -1
    Debug.Print(myArray(i))
Next

Output: 5 4 3 2 1

Upvotes: 1

Jon Skeet
Jon Skeet

Reputation: 1500055

I think the documentation referenced in Mike's answer below is extremely misleading. The order of For Each is defined by the collection it's called (i.e. its implementation of IEnumerable/IEnumerable<T>), but that's not the same as saying it shouldn't be used when the order is important. Many collections (such as arrays, List<T> etc) always go in the "natural" order.

Part of the documentation does allude to this:

Traversal Order. When you execute a For Each...Next loop, traversal of the collection is under the control of the enumerator object returned by the GetEnumerator method. The order of traversal is not determined by Visual Basic, but rather by the MoveNext method of the enumerator object. This means that you might not be able to predict which element of the collection is the first to be returned in element, or which is the next to be returned after a given element.

That's not at all the same as saying it can't be relied upon - it can be relied upon if you know that the collection you're iterating over will produce the results in the desired order. It's not like it's going to pick elements at random. The behaviour in terms of IEnumerable/IEnumerable<T> is clearly defined on that page.

The most important exceptions to predictable orderings are dictionaries and sets, which are naturally unordered.

To reverse an IEnumerable<T>, use Enumerable.Reverse - but if you need to iterate in reverse over a collection which is indexed by position (such as an array or List<T>) then it would be more efficient to use a For loop starting at the end and working backwards.

Upvotes: 47

Georg
Georg

Reputation: 404

' create a list of whatever and step through For Loop Collecting

dim revList as New List (of ToolStripItem)  
For each objItem as ToolStripItem in Menu.DropDownItems
    revList.Add(objItem)
next
revList.reverse ' reverse the order of that list

' step through the reverse ordered list

for each objItem as ToolStripItem in revList   
    ' do whatever (like removing your menu from an ArrayMenu)
next

' replace ToolStripItem and Menu.DropDownItems with whatever you need

Upvotes: 0

Adam
Adam

Reputation: 6733

The accepted answer explains why, but to add another example, for a collection with key (SortedList, SortedDictionary, Dictionary, etc.) you can do

For Each Id as Tkey in MyCollection.Keys.Reverse
   // The item in question is now available as MyCollection(Id)
Next

Upvotes: 0

Hisham Shaaban
Hisham Shaaban

Reputation: 21

first you should create a list(of string) in for each statement, and after that make another normal for .. step -1 ..next statement. see an example:

Dim ff As New List(Of Integer)
For Each rRow As DataRow In DeletedRows
Dim item As Integer
item = rRow.Table.Rows.IndexOf(rRow)
ff.Add(item)
Next
For i As Integer = ff.Count - 1 To 0 Step -1
dim item as integer=ff(i)
next i

i hope that be helpful

Upvotes: 0

user4954249
user4954249

Reputation:

You can add a extended function to the class you are trying to reverse

<Serializable()> Public Class SomeCollection
    Inherits CollectionBase
    Public Sub New()
    End Sub

    Public Sub Add(ByVal Value As Something)
        Me.List.Add(Value)
    End Sub

    Public Sub Remove(ByVal Value As Something)
        Me.List.Remove(Value)
    End Sub

    Public Function Contains(ByVal Value As Something) As Boolean
        Return Me.List.Contains(Value)
    End Function

    Public Function Item(ByVal Index As Integer) As Something
        Return DirectCast(Me.List.Item(Index), Something)
    End Function

    Public Function Reverse() As SomeCollection
        Dim revList As SomeCollection = New SomeCollection()
        For index As Integer = (Me.List.Count - 1) To 0 Step -1
             revList.List.Add(Me.List.Item(index))
        Next
        Return revList
    End Function
End Class

Then you would call it like this

For Each s As Something In SomeCollection.Reverse

Next

Upvotes: 0

Dave
Dave

Reputation: 7359

Testig in Framework 4, the code

For Each s As String In myStringList.Reverse
    //' Do stuff here
Next

it wasn't working, the right way to do it is:

myStringList.Reverse() ' it is a method not a function
For Each s As String In myStringList
//' Do stuff here
Next

Look at: MSDN: Reserve

Upvotes: 4

Tonko Boekhoud
Tonko Boekhoud

Reputation: 961

For Each s As String In myStringList.Reverse
    //' Do stuff here
Next

Upvotes: 3

Amy B
Amy B

Reputation: 110071

Call the System.Linq.Enumerable.Reverse method to get an IEnuemrable(Of TSource) in the reverse order of your enumerable source.

Upvotes: 12

Michael J. Berry
Michael J. Berry

Reputation: 1

Depending on what happens inside your loop you can do an .InsertAt(object,0) instead of an .Add and produce the same result as a reverse enumeration.

Upvotes: 0

xxxmandoxxx
xxxmandoxxx

Reputation: 1

What you have to do is create an array with your for each you had before, then use array.reverse and run the for each on the array. Done

Cheers

Upvotes: 0

reinierpost
reinierpost

Reputation: 8591

The reason it can't be done is that, as a basic design feature, For Each can iterate over an enumerable without a last element, or with an as yet unknown last element.

Upvotes: 1

Mike
Mike

Reputation: 4590

Sadly, the MSDN docs on For Each state that the For Each construct is there explicitly for cases where the order of the iteration is unimportant (unordered sets and the like). So there is unfortunately no concept of reversing a For Each, as the order is not defined anyway.

Good luck!

Upvotes: 12

SqlRyan
SqlRyan

Reputation: 33914

You'd have to do something like:

For i as integer = myStringList.Count-1 to 0 step -1
    dim s as string = myStringList.Item(i)
    ' Do your stuff with s
Next i

But as far as I know, you can't do a "For...Each" backwards, though that would be nice in a few instances.

Upvotes: 46

Related Questions