Performance difference between iterating once and iterating twice?
Asked Answered
A

6

10

Consider something like...

for (int i = 0; i < test.size(); ++i) {
        test[i].foo();
        test[i].bar();
}

Now consider..

for (int i = 0; i < test.size(); ++i) {
        test[i].foo();
}
for (int i = 0; i < test.size(); ++i) {
        test[i].bar();
}

Is there a large difference in time spent between these two? I.e. what is the cost of the actual iteration? It seems like the only real operations you are repeating are an increment and a comparison (though I suppose this would become significant for a very large n). Am I missing something?

Ambroseambrosi answered 24/10, 2010 at 2:50 Comment(0)
R
2

First, as noted above, if your compiler can't optimize the size() method out so it's just called once, or is nothing more than a single read (no function call overhead), then it will hurt.

There is a second effect you may want to be concerned with, though. If your container size is large enough, then the first case will perform faster. This is because, when it gets to test[i].bar(), test[i] will be cached. The second case, with split loops, will thrash the cache, since test[i] will always need to be reloaded from main memory for each function.

Worse, if your container (std::vector, I'm guessing) has so many items that it won't all fit in memory, and some of it has to live in swap on your disk, then the difference will be huge as you have to load things in from disk twice.

However, there is one final thing that you have to consider: all this only makes a difference if there is no order dependency between the function calls (really, between different objects in the container). Because, if you work it out, the first case does:

test[0].foo();
test[0].bar();
test[1].foo();
test[1].bar();
test[2].foo();
test[2].bar();
// ...
test[test.size()-1].foo();
test[test.size()-1].bar();

while the second does:

test[0].foo();
test[1].foo();
test[2].foo();
// ...
test[test.size()-1].foo();
test[0].bar();
test[1].bar();
test[2].bar();
// ...
test[test.size()-1].bar();

So if your bar() assumes that all foo()'s have run, you will break it if you change the second case to the first. Likewise, if bar() assumes that foo() has not been run on later objects, then moving from the second case to the first will break your code.

So be careful and document what you do.

Remember answered 24/10, 2010 at 3:54 Comment(0)
F
1

There are many aspects in such comparison.

First, complexity for both options is O(n), so difference isn't very big anyway. I mean, you must not care about it if you write quite big and complex program with a large n and "heavy" operations .foo() and bar(). So, you must care about it only in case of very small simple programs (this is kind of programs for embedded devices, for example).

Second, it will depend on programming language and compiler. I'm assured that, for instance, most of C++ compilers will optimize your second option to produce same code as for the first one.

Third, if compiler haven't optimized your code, performance difference will heavily depend on the target processor. Consider loop in a term of assembly commands - it will look something like this (pseudo assembly language):

LABEL L1:
          do this    ;; some commands
          call that
          IF condition
          goto L1
          ;; some more instructions, ELSE part

I.e. every loop passage is just IF statement. But modern processors don't like IF. This is because processors may rearrange instructions to execute them beforehand or just to avoid idles. With the IF (in fact, conditional goto or jump) instructions, processors do not know if they may rearrange operation or not.
There's also a mechanism called branch predictor. From material of Wikipedia:

branch predictor is a digital circuit that tries to guess which way a branch (e.g. an if-then-else structure) will go before this is known for sure.

This "soften" effect of IF's, through if the predictor's guess is wrong, no optimization will be performed.

So, you can see that there's a big amount of conditions for both your options: target language and compiler, target machine, it's processor and branch predictor. This all makes very complex system, and you cannot foresee what exact result you will get. I believe, that if you don't deal with embedded systems or something like that, the best solution is just to use the form which your are more comfortable with.

Fadil answered 24/10, 2010 at 3:51 Comment(1)
IIRC, branch prediction usually works really well with loops that don't have likely breaks.Remember
P
0

For your examples you have the additional concern of how expensive .size() is, since it's compared for each time i increments in most languages.

How expensive is it? Well that depends, it's certainly all relative. If .foo() and .bar() are expensive, the cost of the actual iteration is probably minuscule in comparison. If they're pretty lightweight, then it'll be a larger percentage of your execution time. If you want to know about a particular case test it, this is the only way to be sure about your specific scenario.

Personally, I'd go with the single iteration to be on the cheap side for sure (unless you need the .foo() calls to happen before the .bar() calls).

Phidias answered 24/10, 2010 at 2:53 Comment(3)
Assume size() is optimized out. But I just wanted to make sure that the only additional cost is that of an increment and comparison.Ambroseambrosi
And there's the question of how test[i] is optimized in the loops. A single iteration may only require one dereference. But, as always, testing is the best way to answer a performance question.Rosmunda
@Rosmunda - Agreed, and test it at least 3 times itself/overall in a loop (for spool up/down changes), taking the middle result. It depends on the language as to what's necessary.Phidias
T
0

I assume .size() will be constant. Otherwise, the first code example might not give the same as the second one.

Most compilers would probably store .size() in a variable before the loop starts, so the .size() time will be cut down.

Therefore the time of the stuff inside the two for loops will be the same, but the other part will be twice as much.

Tryst answered 24/10, 2010 at 3:3 Comment(0)
E
0

Performance tag, right.

As long as you are concentrating on the "cost" of this or that minor code segment, you are oblivious to the bigger picture (isolation); and your intention is to justify something that, at a higher level (outside your isolated context), is simply bad practice, and breaks guidelines. The question is too low level and therefore too isolated. A system or program which is set of integrated components will perform much better that a collection of isolated components.

The fact that this or that isolated component (work inside the loop) is fast or faster is irrelevant when the loop itself is repeated unnecessarily, and which would therefore take twice the time.

Given that you have one family car (CPU), why on Earth would you:

  • sit at home and send your wife out to do her shopping
  • wait until she returns
  • take the car, go out and do your shopping
  • leaving her to wait until you return If it needs to be stated, you would spend (a) almost half of your hard-earned resources executing one trip and shopping at the same time and (b) have those resources available to have fun together when you get home.

It has nothing to do with the price of petrol at 9:00 on a Saturday, or the time it takes to grind coffee at the café, or cost of each iteration.

Yes, there is a large diff in the time and the resources used. But the cost is not merely in the overhead per iteration; it is in the overall cost of the one organised trip vs the two serial trips.

Performance is about architecture; never doing anything twice (that you can do once), which are the higher levels of organisation; integrated of the parts that make up the whole. It is not about counting pennies at the bowser or cycles per iteration; those are lower orders of organisation; which ajust a collection of fragmented parts (not a systemic whole).

Masseratis cannot get through traffic jams any faster than station wagons.

Ellette answered 24/10, 2010 at 4:38 Comment(2)
You appear to have fixed ideas about the answers that you seek, so fixed that you cannot remove yourself from it long enough to entertain an analogy. Have you heard of the notion that the answer may NOT be what you seek, something BEYOND what you seek ? Are you open to learning about your problem space or just seeking narrow-minded confirmation of an isolated thought ? In order to notice such, if you are earnest, you need to give up your fixed ideas about the answer, and genuinely seek. Fixed minds should not read past the 1st para.Ellette
Separately (ie. a point separate to that which I have already made, that has a bearing on your question), refer Donald Knuth ""Programmers waste enormous amounts of time thinking about, or worrying about, the speed of noncritical parts of their programs ... premature optimization is the root of all evil. "".Ellette
S
0

It does add up. Here is the difference between two cases for the Javascript runtime. It took nearly double amount of time.

However, sometimes I would still loop twice in an interest of better readability and maintainability given that have to loop through small number of elements.

enter image description here

Stover answered 11/7, 2024 at 0:13 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.