Introduction #
This serves as a practical guide to asymptotics. We will not be talking about theory or why asymptotics is neccessary or useful. This guide assumes that you already know those things and want to apply them to find runtimes of actual Java methods. We’ll cover some strategies that you should know and then we’ll go over what I call “cheats,” (warning: there are a lot of cheats and I’ll only be discussing a few).
Without further adiou, let’s start off by talking about some quick math things that you’ll absolutely need to know.
Math background #
There are just a few tiny math things you’ll need to know for asymptotics. One will be simplifying summations, and the other is a quick trick on figuring out how many elements there are in a sequence.
Summations #
Most of asymptotics is really just simplifying the work done into a summation and then applying the right summation formula. The two types of summations you’ll see in CS 61B are the arithmetic summation and the geometric summation.
Arithmetic Summation #
An arithmetic summation is one where the difference between consecutive terms is some constant. The constant doesn’t matter. For example:
are all arithmetic summations.
With an arithmetic summation, the entire summation is
Seemingly hard question: what is the
You might think it’s hard because you see a
Geometric Summation #
Arguably the easier summation. A geometric summation is one where the ratio of consecutive terms is constant. For example:
With a geometric summation, you still take the last term but that’s it. No squaring. Take the last term, throw a
Seemingly hard question: what is the
You’ve probably caught onto the trick. It’s not hard for you now since you’re equipped! We see from the first few terms that the summation is geometric and so we just throw a
Summation summary #
Let’s quickly summarize the summation lessons. If you see a summation, take these 2 steps:
- Look at the first few terms of the summation to see what type it is (arithmetic or geometric).
- Figure out the last term, and then apply the appropriate simplification (if it is arithmetic, square and put a
, and if it is geometric just put a ).
Sequences #
Sequences are slightly different than summations. Instead of summing things, it’s just a list of numbers. The following are sequences:
Notice the commas instead of
There is no concept of “arithmetic” or “geometric” sequence. They’re just sequences. Often we’ll boil the work done in a Java method to a sequence of numbers and we simply want to know how many numbers are here. So here is how I suggest you do that.
You’ll need to know the last term. Let’s say it’s a function of
- Figure out an expression for the
th element in the sequence. Call it . You can make the sequence 0-indexed or 1-indexed, whichever leads to a cleaner expression for . 0-indexed means is the first term, and 1-indexed means is the first term. - Make a new variable called
and set the number of elements in the sequence to . Our objective is to determine . and solve for . Read the above definitions of and if it doesn’t make sense why we do this.
That’s it. Let’s do that for the above 2 sequences.
Oh look, we’re already done. So there are
This one is a bit more complex, though you might be able to do it in your head. Regardless, let’s figure out
And that’s it! Now let’s talk about one sequence that you will see a lot and you might as well just memorize this instead of doing it every time.
Voila. Remember that the base doesn’t matter with
Here is a hard one that I’ll leave for you to do by yourself:
Those might seem like random numbers, but I recommend writing each number as a
power of
And with that, we’ve gone over the mathematical background. You’ll soon see how important these things are. It’s ok if you just memorized what was said so far. But from here on out, you shouldn’t memorize any formulas. Just the strategies. Without further adieu, let’s start with single loop Java methods!
Single loops #
This is the bread and butter of asymptotics questions. You’ll soon see how important those math things we did above were. First let’s look at a few examples of what these Java methods might look like:
public void loop1(int N) {
for (int i = 0; i < N; i = i + 1) {
System.out.println(i);
}
}
This is a pretty standard function. Let’s look at another one.
public void loop2(int N) {
for (int i = 0; i < N*N; i = i + 1) {
int[] x = new int[i];
}
}
One more.
public void loop3(int N) {
for (int i = 1; i < N; i = i * 2) {
int x = i;
}
}
I think we’ve seen enough. All of these look basically the same. There isn’t much more to change here. Don’t worry about finding the runtimes of those things above, we’ll do that soon.
Just 3 things changed across all of those functions:
- The exit condition of the loop. Say the biggest value
can be is . This means our exit condition is . - How
increments every iteration as a function of the previous . - How much work gets done inside the loop as a function of
. Call this function .
With these things, we can quickly find the runtime of the function by making a summation:
Here,
You’ll find these functions
That’s it. Let’s revisit loop1
and see how to actually do this.
Loop 1 #
public void loop1(int N) {
for (int i = 0; i < N; i = i + 1) {
System.out.println(i);
}
}
is the exit condition, so is the increment function- ???
I hope you see how we got the first 2. I just looked at the for-loop and it’s those last 2 things. Pretty simple right?
What about how much work gets done as a function of System.out.println(i)
is
Ok, now we’re ready for the next step: making the summation.
We know that
We know that
This summation is super easy. It’s just
Important thing to notice: if the work done per iteraiton is constant with respect to
Let’s do loop2
which is slightly more complicated.
Loop 2 #
public void loop2(int N) {
for (int i = 0; i < N*N; i = i + 1) {
int[] x = new int[i];
}
}
is the exit condition, so is the increment function- ???
Again, the first 2 aren’t too bad to get. Just reading the code.
What about how much work gets done as a function of
Ok, now we’re ready for the next step: making the summation.
We know that
We know that
Does this summation look familiar? It should. We look at the first few terms and determine it’s arithmetic, so we take the last term, square it, and put
Finally, let’s do loop3
.
Loop 3 #
public void loop3(int N) {
for (int i = 1; i < N; i = i * 2) {
int x = i;
}
}
is the exit condition, so is the increment function- ???
Again, the first 2 aren’t too bad to get. Just reading the code.
What about how much work gets done as a function of
Now here we could continue with the rest of the problem, or remember what I said above. I’ll repeat it.
Important thing to notice: if the work done per iteraiton is constant with respect to
, you can simply count the number of iterations and put on that number.
Let’s generate a sequence of the values of
We’re basically done. I did this above in the previous section. We know there are
Sumary of single loop Java methods #
We learned a strategy for approaching these.
- Figure out the last value of
as a function of . Call this . - Figure out the increment function of
. - Figure out the work done per iteration as a function of
. Call this .
Once you have the above, you simply write out the summation as:
You figure out what
We also saw that if
After a few of these problems, you’ll likely be able to look at a single loop and instantly tell what its runtime is. That’s good!
You might think that I overcomplicated things here. That there are shortcuts. And you’re right. There are shortcuts. But there are so many shortcuts. There is no way I could possibly detail every shortcut you can take in this guide, nor is it that helpful for you as someone learning asymptotics. The goal of this is to give you strategies to figure out the answer yourself. You’ll see in the next section when we do double loops that this method pays off and is really easy to extend.
You should try a few functions before moving onto the next section. Try making your own functions with different
and you can just leave it like that. I don’t know how to simplify that and it’s out of the scope of this class.
Double loops #
I made this a separate section but really it’s no different than single loops. It is a bit more involved, so I want to go over them with you. Firstly, let me show you a few examples of double loops:
public void dLoop1(int N) {
for (int i = 1; i < N; i = i * 2) {
for (int j = 0; j < 10; j = j + 1) {
System.out.println(i + j);
}
}
}
Pretty standard. Let’s check out another one.
public void dLoop2(int N) {
for (int i = 1; i < N; i = i + 1) {
for (int j = 1; j < i; j = j * 2) {
int[] x = new int[j];
}
}
}
And one more:
public void dLoop3(int N) {
for (int i = 1; i < N * N; i = i + 1) {
for (int j = 1; j < Math.sqrt(i); j = j + 1) {
int[] x = new int[j];
}
}
}
Firstly, do not fall down the oldest trick in the book. Do not multiply the number of iterations of the outer loop by the number of iteration of the inner loop. That’s almost never what you’re supposed to do. It’s a common mistake many students make because they try taking a shortcut. Shortcuts are nice, but don’t use them unless you’re sure of how to use it!!
With that out of the way, let’s talk about how you do these. It’s literally the same as single loops:
- Figure out the last value of
as a function of . Call this . - Figure out the increment function of
. - Figure out the work done per iteration as a function of
. Call this . Once you have the above, you simply write out the summation as:
You figure out what are by looking at the increment function for . Then, plug in everything you know and then simplify the sum. Voila, that’s it.
Here,
So, let’s make steps for ourselves:
- Figure out the work done of the inner loop as a function of
. Call this . - Take this
, and use it to find the work the outer loop does. Put on it.
Note that
That’s it. Let’s see it in action with dLoop1
Double Loop 1 #
public void dLoop1(int N) {
for (int i = 1; i < N; i = i * 2) {
for (int j = 0; j < 10; j = j + 1) {
System.out.println(i + j);
}
}
}
Step 1 is to find the work done of the inner loop as a function of
for (int j = 0; j < 10; j = j + 1) {
System.out.println(i + j);
}
Do you notice anything peculiar about the inner loop? It has an
Now, we see that it’s basically the same as loop3
above, and we get a final answer of
Double Loop 2 #
public void dLoop2(int N) {
for (int i = 1; i < N; i = i + 1) {
for (int j = 1; j < i; j = j * 2) {
int[] x = new int[j];
}
}
}
A bit more complicated. Now
Remember, we just isolate ourselves to the inner loop.
for (int j = 1; j < i; j = j * 2) {
int[] x = new int[j];
}
And we want to find the work done of this inner loop as a function of
This is simply a single loop! We know how to do those! I’ll skip to the last step, but you should do all the steps to test your understanding and get more practice.
You’ll get this summation:
We know this summation is
But I said not to take a
This is where things get a little bit weird.
We can pretend that the work done is
evaluates to
So
Now we do the single loop stuff and get this summation:
You’ve seen this many times now. It’s simply
So dloop2
has an asymptotic runtime of
Back to the weirdness of earlier. We said that since
Double Loop 3 #
Final one:
public void dLoop3(int N) {
for (int i = 1; i < N * N; i = i + 1) {
for (int j = 1; j < Math.sqrt(i); j = j + 1) {
int[] x = new int[j];
}
}
}
Step 1 is to find the work done of the inner loop as a function of
for (int j = 1; j < Math.sqrt(i); j = j + 1) {
int[] x = new int[j];
}
EDIT: in an earlier (wrong) version of this guide, the update was instead j = j * 2
which lead to a weird sum that you’re not expected to know how to simplify (I don’t know how to simplify it either).
I’ll again skip the steps and go straight to the summation, but you should do it from the start. Practice makes better.
Which we know is
Now, with the outer loop we’ll use this function
This is an arithmetic summation, so we take the last term, square it, and throw a dLoop3
has a running time that is
I hope you see that this is really not that bad. It’s just sticking to the rules.
Summary of double loop Java methods #
We learned a strategy for approaching these.
- Figure out the work done in the inner loop as a function of
. Call this . - Use this
with the outer loop and the same exact process
Unlike singe loops, you’ll likely not be able to look at a double loop and instantly tell what its runtime is. That’s OK. Whenever I do double loops I always write it out since it’s so easy to make. a mistake if you try to do it in your head. The process is tried and tested, so just practice using it and you’ll always get these ones right.
You should now appreciate the way we did single loops. It made double loops so much easier. You could even do triple, quadruple, and more loops. But … that’s super annoying. I’m not saying it’s out of scope for any exam since it’s basically the same thing as a double loop, I’m just saying it’s annoying.
You should try a few functions before moving onto the next section. Just like with single loops, try making your own functions with different
That’s it for nested loop asymptotics questions. You are now certified to do loop problems and teach your friends too as that is the absolute best way to learn. Next up is recursion!