Loops are often used in programs that compute numerical results by starting with an approximate answer and iteratively improving it.
For example, one way of computing square roots is the Babylonian method. Suppose that you want to know the square root of . If you start with almost any estimate, , you can compute a better estimate with the following formula:
For example, if is 4 and is 3:
The result is closer to the correct answer (). If we repeat the process with the new estimate, it gets even closer:
After a few more updates, the estimate is almost exact:
In general we don’t know ahead of time how many steps it takes to get to the right answer. But we can tell when it gets there, because the square of is
When g**2 == a, we can stop. But another way, we continue while g͡**2 is not a. Here is a loop that starts with an initial estimate, x, and improves it until it stops changing:
For most values of a this works fine, but in general it is dangerous to test float equality. For example, this code snippet will not work when . Floating-point values are only approximately right: most rational numbers, like , and irrational numbers, like , can’t be represented exactly with a float.
Rather than checking whether g**2 and a are exactly equal, it is safer to use the built-in function abs to compute the absolute value, or magnitude, of the difference between them. Here we repeat while the “error” is greater than some constant epsilon.
Where epsilon
has a value like 0.0000001 that
determines how close is close enough.
The Babylonian method is an example of an algorithm: it is a mechanical process for solving a category of problems (in this case, computing square roots).
To understand what an algorithm is, it might help to start with something that is not an algorithm. When you learned to multiply single-digit numbers, you probably memorized the multiplication table. In effect, you memorized 100 specific solutions. That kind of knowledge is not algorithmic.
But if you were “lazy”, you might have learned a few tricks. For example, to find the product of and 9, you can write as the first digit and as the second digit. This trick is a general solution for multiplying any single-digit number by 9. That’s an algorithm!
Similarly, the techniques you learned for addition with carrying, subtraction with borrowing, and long division are all algorithms. One of the characteristics of algorithms is that they do not require any intelligence to carry out. They are mechanical processes where each step follows from the last according to a simple set of rules.
Executing algorithms is boring, but designing them is interesting, intellectually challenging, and a central part of computer science.
Some of the things that people do naturally, without difficulty or conscious thought, are the hardest to express algorithmically. Understanding natural language is a good example. We all do it, but so far no one has been able to explain how we do it, at least not in the form of an algorithm.
As you start writing bigger programs, you might find yourself spending more time debugging. More code means more chances to make an error and more places for bugs to hide.
One way to cut your debugging time is “debugging by bisection”. For example, if there are 100 lines in your program and you check them one at a time, it would take 100 steps.
Instead, try to break the problem in half. Look at the middle of the program, or near it, for an intermediate value you can check. Add a print statement (or something else that has a verifiable effect) and run the program.
If the mid-point check is incorrect, there must be a problem in the first half of the program. If it is correct, the problem is in the second half.
Every time you perform a check like this, you halve the number of lines you have to search. After six steps (which is fewer than 100), you would be down to one or two lines of code, at least in theory.
In practice it is not always clear what the “middle of the program” is and not always possible to check it. It doesn’t make sense to count lines and find the exact midpoint. Instead, think about places in the program where there might be errors and places where it is easy to put a check. Then choose a spot where you think the chances are about the same that the bug is before or after the check.
Assigning a new value to a variable that already exists.
An assignment where the new value of the variable depends on the old.
An assignment that gives an initial value to a variable that will be updated.
An update that increases the value of a variable (often by one).
An update that decreases the value of a variable.
Repeated execution of a set of statements using either a recursive function call or a loop.
A loop in which the terminating condition is never satisfied.
A general process for solving a category of problems.
Copy the loop from Section 13
and encapsulate it in a function called
mysqrt
that takes a as a parameter, chooses a
reasonable value of x, and returns an estimate of the square
root of a.
To test it, write a function named test_square_root
that prints a table like this:
The first column is a number, ; the second column is the square
root of computed with mysqrt
; the third column is the
square root computed by math.sqrt; the fourth column is the
absolute value of the difference between the two estimates.
The mathematician Srinivasa Ramanujan found an infinite series that can be used to generate a numerical approximation of :
Write a function called estimate_pi
that uses this formula
to compute and return an estimate of . It should use a while
loop to compute terms of the summation until the last term is
smaller than 1e-15 (which is Python notation for ).
You can check the result by comparing it to math.pi.