2.4. An Anagram Detection ExampleΒΆ
A good example problem for showing algorithms with different orders of
magnitude is the classic anagram detection problem for strings. One
string is an anagram of another if the second is simply a rearrangement
of the first. For example, heart
and earth
are anagrams. The
strings python
and typhon
are anagrams as well. For the sake
of simplicity, we will assume that the two strings in question are of
equal length and that they are made up of symbols from the set of 26
lowercase alphabetic characters. Our goal is to write a boolean function
that will take two strings and return whether they are anagrams.
2.4.1. Solution 1: Anagram Detection Checking OffΒΆ
Our first solution to the anagram problem will check the lengths of the
strings and then check to see that each character in the first string actually
occurs in the second. If it is possible to check off each character, then
the two strings must be anagrams. Checking off a character will be
accomplished by replacing it with the special Python value None
.
However, since strings in Python are immutable, the first step in the
process will be to convert the second string to a list. Each character
from the first string can be checked against the characters in the list
and if found, checked off by replacement. ActiveCode 1 shows this function.
To analyze this algorithm, we need to note that each of the n
characters in s1
will cause an iteration through up to n
characters in the list from s2
. Each of the n positions in the
list will be visited once to match a character from s1
. The number
of visits then becomes the sum of the integers from 1 to n. We stated
earlier that this can be written as
As
2.4.2. Anagram Detection Solution 2: Sort and CompareΒΆ
Another solution to the anagram problem will make use of the fact that
even though s1
and s2
are different, they are anagrams only if
they consist of exactly the same characters. So if we begin by sorting
each string alphabetically from a to z, we will end up with the same
string if the original two strings are anagrams. ActiveCode 2 shows
this solution. Again, in Python we can use the built-in sort
method
on lists by simply converting each string to a list at the start.
At first glance you may be tempted to think that this algorithm is
sort
method are not without their own cost. As we will see in
Chapter 5, sorting is typically either
2.4.3. Anagram Detection Solution 3: Brute ForceΒΆ
A brute force technique for solving a problem typically tries to
exhaust all possibilities. For the anagram detection problem, we can
simply generate a list of all possible strings using the characters from
s1
and then see if s2
occurs. However, there is a problem
with this approach. When generating all possible strings from s1
,
there are n possible first characters,
It turns out that s1
were 20 characters long, there would
be
2.4.4. Anagram Detection Solution 4: Count and CompareΒΆ
Our final solution to the anagram problem takes advantage of the fact that any two anagrams will have the same number of aβs, the same number of bβs, the same number of cβs, and so on. In order to decide whether two strings are anagrams, we will first count the number of times each character occurs. Since there are 26 possible characters, we can use a list of 26 counters, one for each possible character. Each time we see a particular character, we will increment the counter at that position. In the end, if the two lists of counters are identical, the strings must be anagrams. ActiveCode 3 shows this solution.
Again, the solution has a number of iterations. However, unlike the
first solution, none of them are nested. The first two iterations used
to count the characters are both based on n. The third iteration,
comparing the two lists of counts, always takes 26 steps since there are
26 possible characters in the strings. Adding it all up gives us
Before leaving this example, we need to say something about space requirements. Although the last solution was able to run in linear time, it could only do so by using additional storage to keep the two lists of character counts. In other words, this algorithm sacrificed space in order to gain time.
This is a common occurrence. On many occasions you will need to make decisions between time and space trade-offs. In this case, the amount of extra space is not significant. However, if the underlying alphabet had millions of characters, there would be more concern. As a computer scientist, when given a choice of algorithms, it will be up to you to determine the best use of computing resources given a particular problem.
Self Check
- O(n)
- In an example like this you want to count the nested loops. especially the loops that are dependent on the same variable, in this case, n.
- O(n^2)
- A singly nested loop like this is O(n^2)
- O(log n)
- log n typically is indicated when the problem is iteratvely made smaller
- O(n^3)
- In an example like this you want to count the nested loops. especially the loops that are dependent on the same variable, in this case, n.
Q-4: Given the following code fragment, what is its Big O running time?
test = 0
for i in range(n):
for j in range(n):
test = test + i * j
- O(n)
- Even though there are two loops they are not nested. You might think of this as O(2n) but we can ignore the constant 2.
- O(n^2)
- Be careful, in counting loops you want to make sure the loops are nested.
- O(log n)
- log n typically is indicated when the problem is iteratvely made smaller
- O(n^3)
- Be careful, in counting loops you want to make sure the loops are nested.
Q-5: Given the following code fragment what is its Big O running time?
test = 0
for i in range(n):
test = test + 1
for j in range(n):
test = test - 1
- O(n)
- Look carefully at the loop variable i. Notice that the value of i is cut in half each time through the loop. This is a big hint that the performance is better than O(n)
- O(n^2)
- Check again, is this a nested loop?
- O(log n)
- The value of i is cut in half each time through the loop so it will only take log n iterations.
- O(n^3)
- Check again, is this a nested loop?
Q-6: Given the following code fragment what is its Big O running time?
i = n
while i > 0:
k = 2 + 2
i = i // 2