I am trying to find the Time Complexity of this algorithm.
The iterative: algorithm produces all the bit-strings within a given Hamming distance, from the input bit-
Notice, that given n
which represents the length, and t
which represents the distance required, the number of increasing, non-negative series of t
integers between 1
and n
(or in indices form, between 0
and n-1
) is indeed n choose t
, since we pick t
distinct indices.
The problem occurs with your generation of those series:
-First, notice that for example in the case of length 4, you actually go over 5 different indices, 0 to 4.
-Secondly, notice that you are taking in account series with identical indices (in the case of t=2
, its 0 0, 1 1, 2 2
and so on), and generally, you would go through every non-decreasing series, instead of through every increasing series.
So for calculating the TC of your program, make sure you take that into account.
Hint: try to make one-to-one correspondence from the universe of those series, to the universe of integer solutions to some equation.
If you need the direct solution, take a look here : https://math.stackexchange.com/questions/432496/number-of-non-decreasing-sequences-of-length-m
The final solution is (n+t-1) choose (t)
, but noticing the first bullet, in your program, its actually ((n+1)+t-1) choose (t)
, since you loop with one extra index.
Denote
((n+1)+t-1) choose (t) =: A
, n choose t =: B
overall we get O(1) + B*O(n) + (A-B)*O(1)
The while loop is somewhat clever and subtle, and it's arguable that it's doing two different things (or even three if you count the initialisation of a
). That's what's making your complexity calculations challenging, and it's also less efficient than it could be.
In the abstract, to incrementally compute the next set of indices from the current one, the idea is to find the last index, i
, that's less than n-dist+i
, increment it, and set the following indexes to a[i]+1
, a[i]+2
, and so on.
For example, if dist=5, n=11 and your indexes are:
0, 3, 5, 9, 10
Then 5
is the last value less than n-dist+i
(because n-dist
is 6, and 10=6+4, 9=6+3, but 5<6+2).
So we increment 5
, and set the subsequent integers to get the set of indexes:
0, 3, 6, 7, 8
Now consider how your code runs, assuming k=4
0, 3, 5, 9, 10
a[k] + 1
is 11, so k
becomes 3.++a[k]
is 10, so a[k+1]
becomes 10, and k
becomes 4.++a[k]
is 11, so k
becomes 3.++a[k]
is 11, so k
becomes 2.++a[k]
is 6, so a[k+1]
becomes 6, and k
becomes 3.++a[k]
is 7, so a[k+1]
becomes 7, and k
becomes 4.++a[k]
is 8, and we continue to call the print
function.This code is correct, but it's not efficient because k
scuttles backwards and forwards as it's searching for the highest index that can be incremented without causing an overflow in the higher indices. In fact, if the highest index is j
from the end, the code uses a non-linear number iterations of the while loop. You can easily demonstrate this yourself if you trace how many iterations of the while loop occur when n==dist
for different values of n
. There is exactly one line of output, but you'll see an O(2^n) growth in the number of iterations (in fact, you'll see 2^(n+1)-2 iterations).
This scuttling makes your code needlessly inefficient, and also hard to analyse.
Instead, you can write the code in a more direct way:
void hamming2(const char* num, size_t dist) {
int a[dist];
for (int i = 0; i < dist; i++) {
a[i] = i;
}
size_t n = strlen(num);
while (true) {
print(num, a);
int i;
for (i = dist - 1; i >= 0; i--) {
if (a[i] < n - dist + i) break;
}
if (i < 0) return;
a[i]++;
for (int j = i+1; j<dist; j++) a[j] = a[i] + j - i;
}
}
Now, each time through the while loop produces a new set of indexes. The exact cost per iteration is not straightforward, but since print
is O(n), and the remaining code in the while loop is at worst O(dist), the overall cost is O(N_INCR_SEQ(n, dist) * n), where N_INCR_SEQ(n, dist) is the number of increasing sequences of natural numbers < n of length dist. Someone in the comments provides a link that gives a formula for this.