Wall of text time.
IMO, if you're not VSyncing... you should
always be sleeping/delaying. Sucking up 100% CPU is completely unnecessary and if the poor soul is running the game on their laptop it will suck all the juice out of their battery.
That aside... let's take a look at another problem here... and examine possible solutions.
One problem which hasn't been mentioned yet is that variable update periods can produce varying/inconsistent in-game behavior. For example... collisions are usually checked only once per update regardless of the time period of that update.
For example.. if there's a collision that's occuring within a 20 ms time period... if you process that time period in one 20 ms update() you will only register 1 collision.. whereas if you are processing every 5 ms you will register 4 collisions (once each update). This can impact collision logic, movement logic, etc. In extreme [albeit unlikely] cases, it can even result in game-breaking behavior (like tunneling) if you're not careful.
In any event this can lead to people who are running on faster machines having a different in-game experience than people who are running on slower machines... even if the slower machines are getting a full framerate with no slowdown.
This also makes it extremely difficult to have strictly deterministic behavior (if, for example you want to be able to record in-game movies by recording keypresses... or do 'Braid' style rewind effects) since the CPU time your program gets becomes an influencing factor in game logic.
The easiest way to solve this problem is to use a fixed-time update. IE... to say that one logic update occurs every X milliseconds. Back in the day.... it was very common for this timeframe to be equal to the length of one frame. So if you were drawing 60 FPS... you would also update logic at 60 FPS.
This presents another problem if your logic rate does not match the user's monitor refresh rate. What if the user's monitor updates at 75 FPS instead of 60? Then VSyncing is not an option or else your logic will run too fast. This is less of a problem now than it was when CRTs were common... but still.
To solve both of these problems I meet somewhere in the middle. Fixed logic rate... but let the display have whatever framerate it needs. In a recent project I had logic updating at 50 updates per second, while graphics (usually) displayed at 60 FPS. 50 UPS makes a nice and round 20 ms per logic update... not a messy 16.666667... and 50 UPS is certainly fast enough for the program to feel responsive to the user.
So the glaring question here is... if you're updating at 50 UPS but drawing at 60 FPS... wouldn't that look crappy? Wouldn't it make the graphics jittery due to the update rate being slower than the display rate?
The answer: only if you don't do it right.
The trick here is to run the logic 1 update ahead of where it should be... and interpolate the graphic representation of animations and positions of objects between the last 2 physical positions. I know that isn't very clear.... so here's an example.
Let's assume you have an object that is moving at 10 pixels per update (one update every 20 ms). This means that at timestamp 0 he will be at position 0... at timestamp 20 (ms) he will be at position 10, etc, etc:
1 2 3 4 5 6 7 8 9 10
|
LOGIC: (50 UPS)
______________
Time Phys Pos
0 0
20 10
40 20
60 30
80 40
100 50
120 60
| |
The naive approach to this would simple draw the object at its current position whenever the rendering code triggers. If we are drawing at 60 FPS... this produces very ugly results:
1 2 3 4 5 6 7 8 9 10 11 12
|
GRAPHIC: (60 FPS)
__________________
Time Pos (bad)
0 0
16.67 0
33.33 10
50 20
66.67 30
83.33 40
100 50
116.67 50
| |
As you can see positions 0 and 50 are drawn twice. This creates very 'jerky' movement that is unpleasant.
So what I'm proposing is that you run the logic 1 update ahead of where it's supposed to be... then interpolate or "blend" the last two positions together based on where in that 20 ms window the drawing actually happens. So what you get is this:
1 2 3 4 5 6 7 8 9 10 11 12
|
GRAPHIC: (60 FPS)
__________________
Time Pos (good)
0 0
16.67 8
33.33 16
50 25
66.67 33
83.33 41
100 50
116.67 58
| |
So at timestamp 16.6667 you would draw the object at position 8... even though it never actually was at position 8. This can be easily calculated:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
|
// assume prev_pos and next_pos are the two logical/physical positions of the object that
// we are interpolating.
// assume 't' is the timestamp at which the graphics are being rendered.
temp = t % 20; // mod by 20 (because there are 20 ms per logic update)
interpolate = temp / 20.0; // divide by 20 (with floating point math). This is now
// the scale by which to interpolate our 2 values. ie:
// interpolate= 0: draw at prev_pos
// interpolate= 1: draw at next_pos
// interpolate=0.5: draw halfway between prev_pos and next_pos
temp = (next_pos - prev_pos);
pos = prev_pos + (temp * interpolate);
// 'pos' is now the position at which to draw this object
| |
With this... you get
extremely smooth graphics no matter what FPS you're rendering at. Also, your logic only has to run at 50 UPS, which means you can sleep/delay and ease some burden off the CPU.