The Euclidean Algorithm
implementing one of the earliest algorithms in recorded history
Recall that a rational number Fraction
class we will develop in lecture.
Greatest common divisors
Given integers
Given two integers
Example. Consider
Note. For any integer
Computing the gcd
In school, I was taught the following method for computing the GCD of two numbers:
- factor both numbers as a product of prime numbers
- find all of the common factors (with multiplicities)
- the GCD is the product of the common factors
Using the example above with
and
so the common factors are 2 and 3 with multiplicity 1 (2 only appears once in the factorization of 18, and 3 only appears once in the factorization of 24). Thus we have (correctly) found that
The problem with this method of computing the GCD is that you first must factor both numbers. Factoring is fine for small numbers, but once we look at larger numbers, factoring becomes inefficient. In fact, it is unknown whether or not there is an efficient algorithm for factoring large numbers (at least for classical, i.e., non-quantum, computers).
Thankfully, there is an older and more efficient way of computing GCDs: the Euclidean algorithm. To understand the Euclidean algorithm, we must start with long division. Recall that given two integers
In Java, we can compute q = a / b
and r = a % b
. Now this brings us to an important fact:
Important Fact. Suppose
Why does this fact help us? Well, if we know that
But how do we compute
Continuing this process, the GCD remains the same, but the second number continues to decrease. Since we get a decreasing sequence of non-negative integers, eventually (actually, quite quickly) we get a remainder of 0, so we find
Example. Using the example
and
Since
Bigger example. Let’s compute
So putting it all together, we get
The process above defines the Euclidean Algorithm. A bit more formally, we can write the process recursively as follows:
:- if
, return - otherwise, find
and with with - return
- if
Implementing the Euclidean Algorithm
The recursive description of the Euclidean Algorithm above lends itself to a deceptively simple implementation in Java:
1
2
3
4
5
6
7
public int gcd(int a, int b) {
if (b == 0) {
return a;
}
return gcd(b, a % b);
}
This code is both concise and efficient.
IYI: Proving the Important Fact
Here, we prove the important fact from before:
Important Fact. Suppose
Proof. It suffices to show that
Solving this expression for
In particular, this shows that
Now assume that
Therefore,
We have shown that (1) any common divisor of