Skip to main content

๐ŸŽข Java Loops: Fun, Fear, and ForEach() Fails

๐ŸŒ€ Oops, I Looped It Again! — The Ultimate Java Loop Guide You Won't Forget

“I remember this question from one of my early interviews — I was just 2 years into Java and the interviewer asked, ‘Which loop do you prefer and why?’”
At first, I thought, “Duh! for-each is cleaner.” But then he grilled me with cases where it fails. ๐Ÿ˜ต
That led me to explore all loop types, their powers, and their pitfalls.

Let’s deep-dive into every major Java loop with examples & real-world guidance so you'll never forget again.


๐Ÿ” Loop Type #1: Classic For Loop — “The Old Reliable”

✅ When to Use:
  • You need an index
  • You want to iterate in reverse
  • You want full control over loop mechanics
✅ Good Example:
List<String> names = List.of("A", "B", "C");

for (int i = 0; i < names.size(); i++) {

    System.out.println(i + ": " + names.get(i));

}

๐Ÿ”ฅ Reverse + Removal Example:
List<String> items = new ArrayList<>(List.of("X", "Y", "Z"));

for (int i = items.size() - 1; i >= 0; i--) {

    items.remove(i);

}

System.out.println(items); // [] ✅ Safe

❌ Common Pitfall:
for (int i = 0; i < list.size(); i++) {

    list.remove(i); // ❌ May skip elements due to index shift

}

๐Ÿ˜ต While reading about this, I had a doubt — “If this doesn't throw ConcurrentModificationException, why is it still bad?”
✅ Answer: Because removal shifts the elements left, so the loop skips over the next item.

๐Ÿ‘ Loop Type #2: Enhanced ForEach Loop — “The Clean Reader with Boundaries”

✅ When to Use:
  • You want to read values only
  • No need for index or mutation
✅ Good Example:
for (String fruit : fruits) {

    System.out.println(fruit);

}

❌ Bad Example:
for (String name : list) {

    list.remove(name); // ❌ CME will occur

}

๐Ÿ”ฅ Behind the scenes: This uses an implicit iterator. Modifying the list causes CME.
๐Ÿ’ฌ My thought: “So if this is just syntax sugar for an iterator, why not use iterator directly and control it?”
๐Ÿ”ฅ Answer: Exactly! If you plan to remove or modify items, switch to Iterator.

๐Ÿฅท Loop Type #3: Iterator — “The Safe Mutator Ninja”

✅ When to Use:
  • Need to remove elements safely
  • No index required
✅ Safe Removal Example:
Iterator<String> it = list.iterator();

while (it.hasNext()) {

    if (it.next().equals("X")) {

        it.remove(); // ✅ Safe!

    }

}

๐Ÿ”ฅ Internal Magic:
  • modCount tracks list changes
  • expectedModCount is stored inside iterator
  • Mismatch = ๐Ÿ’ฃ ConcurrentModificationException
❌ Unsafe Removal Example:
Iterator<String> it = list.iterator();

while (it.hasNext()) {

    if (it.next().equals("B")) {

        list.remove("B"); // ❌ CME

    }

}

๐Ÿค” I asked Myself “Why does iterator.remove() work but list.remove() doesn’t?”
✅ Answer: Because iterator updates both modCount and expectedModCount in sync!

๐Ÿ”„ Loop Type #4: ListIterator — “The Bi-directional Beast”

✅ When to Use:
  • Need to go forward AND backward
  • Need to add/replace during iteration
✅ Add While Iterating:
ListIterator<String> it = list.listIterator();

while (it.hasNext()) {

    if (it.next().equals("Cat")) {

        it.add("Tiger"); // ✅ Adds after \"Cat\"

    }

}

✅ Replace with set():
while (it.hasNext()) {

    if (it.next().equals("Old")) {

        it.set("New"); // ✅ Replace

    }

}

๐Ÿ” Reverse Loop:
ListIterator<String> it = list.listIterator(list.size());

while (it.hasPrevious()) {

    System.out.println(it.previous());

}

๐Ÿ’ก I wondered: “Can’t we use iterator() and go reverse?”
❌ Answer: Nope! Only ListIterator supports hasPrevious() + previous().

๐ŸŽง Loop Type #5: Stream forEach() — “The Stylish But Stubborn Streamer”

✅ When to Use:
  • Need short, clean, read-only loop
  • Want to chain filter/map/forEach
✅ Good Use:
list.stream()

    .filter(item -> item.length() > 3)

    .map(String::toUpperCase)

    .forEach(System.out::println);

❌ Don't Mutate Inside:
list.forEach(item -> {

    if (item.equals("B")) {

        list.remove(item); // ❌ CME

    }

});

๐Ÿคฏ I asked: “Even stream forEach doesn’t allow modification?!”
✅ Yes, because it uses internal iteration. You're not supposed to mutate the collection during streaming.

๐Ÿง  Final Summary: Which Loop to Use When?

Loop TypeIndexRemoveAddReverseModern?
For Loop
For-Each
Iterator⚠️
ListIterator
Stream.forEach✅✅

๐Ÿคน Wrapping Up — Loop Like a Legend

When someone again asks, “Which loop do you prefer?” — don’t just say “forEach is clean.” ๐Ÿ˜…
Instead, drop a mini-masterclass with examples, trade-offs, and the right loop for the job.

๐Ÿš€ Because Java isn’t just about writing code — it’s about knowing why something works the way it does.

๐Ÿ“ฃ If this helped or made you smile — do share, drop your feedback, and tell me:
๐Ÿงฉ “Which loop got you once in production or an interview?” ๐Ÿ˜๐Ÿ‘‡

Comments

Popular posts from this blog

๐Ÿ” Is final Really Final in Java? The Truth May Surprise You ๐Ÿ˜ฒ

๐Ÿ’ฌ “When I was exploring what to do and what not to do in Java, one small keyword caught my eye — final . I thought it meant: locked, sealed, frozen — like my fridge when I forget to defrost it.”   But guess what? Java has its own meaning of final… and it’s not always what you expect! ๐Ÿ˜… Let’s break it down together — with code, questions, confusion, jokes, and everything in between. ๐ŸŽฏ The Confusing Case: You Said It's Final... Then It Changed?! ๐Ÿซ  final List<String> names = new ArrayList <>(); names.add( "Anand" ); names.add( "Rahul" ); System.out.println(names); // [Anand, Rahul] ๐Ÿคฏ Hold on... that’s final , right?! So how on earth is it still changing ? Time to dive deeper... ๐Ÿง  Why Is It Designed Like This? Here’s the key secret: In Java, final applies to the reference , not the object it points to . Let’s decode this like a spy mission ๐Ÿ•ต️‍♂️: Imagine This: final List<String> names = new ArrayList <>(); Be...

๐ŸŒŸ My Journey – From Zero to Senior Java Tech Lead ๐ŸŒŸ

 There’s one thing I truly believe… If I can become a Java developer, then anyone in the world can. ๐Ÿ’ฏ Sounds crazy? Let me take you back. ๐Ÿ•“ Back in 2015… I had zero coding knowledge . Not just that — I had no interest in coding either. But life has its own plans. In 2016, I got a chance to move to Bangalore and joined a Java course at a training center. That’s where it all started — Every day, every session made me feel like: "Ohhh! Even I can be a developer!" That course didn’t just teach Java — it gave me confidence . ๐Ÿงช Two Life-Changing Incidents 1️⃣ The Interview That Wasn't Planned Halfway through my course, I had to urgently travel to Chennai to donate blood to a family member. After that emotional rollercoaster, I found myself reflecting on my skills and the future. The next day, as I was preparing for my move to Bangalore to complete the remaining four months of my course, I randomly thought — "Let me test my skills... let me just see...