我正在努力学习如何找到各种算法的大角度范围,但我很难理解如何做到这一点,即使在这里阅读了一些问题以及有关该主题的讲座和教科书之后。例如,
procedure for array a{
step=1
while(step <= n){
i = n
while(i >= step){
a[i]= a[i-step] + a[i]
i = i - 1}
step = step * 2}
}
我想弄清楚n在数组a中的索引数量方面的增加数量上的big-theta界限。我可以看到外部循环本身经历了log(n)次迭代,但我无法弄清楚如何表达内部循环中发生的事情。有没有人有解释,或者甚至是我可能会尝试咨询的资源?
答案 0 :(得分:5)
Big Theta 符号要求我们找到2个常数,k1和k2,使得我们的函数f(n)在k1 * g(n)和k2 * g(n)之间,足够大n 。换句话说,我们能找到一些其他函数g(n),它在某个点上小于f(n),在另一个点上大于f(n)(单调地每个方向)。
为了证明Big-Theta,我们需要找到g(n)使得f(n)是O(g(n))(上界),而f(n)是Big-Omega(g(n) )(下界)。
就 Big-O 表示法而言(其中f(n)&lt; = g(n)* k),您的算法f(n)为O(log(n)* n),在这种情况下g(n)= log(n)* n。
我们来证明这一点:
外循环执行多少次?跟踪“步骤”变量: 假设n是100:
对于100的输入,这是7次执行。我们可以等效地说它执行(log n)
次(实际上是floor(log n)次,但是log(n)就足够了。)
现在让我们看一下内循环。跟踪i变量,该变量从n
开始并递减,直到其大小为step
每次迭代。因此,对于步骤的每个值,内部while循环将执行n步步骤。
例如,当n = 100
那么内循环的总迭代次数是多少?
这件事怎么会增长?好吧,因为我们知道外循环会迭代log(n)次,所以我们可以将这个东西表示为求和:
求和(从i = 0到log(n))n-2 ^ i
= log(n)* n - 总和(从i = 0到log(n))2 ^ i
= log(n)* n - (2 ^ 0 + 2 ^ 1 + 2 ^ 2 + ... + 2 ^ log(n))
= log(n)* n - ((1-2 ^ log(n))/(1-2))(实际上2 ^ log(n + 1)但足够接近)
= log(n)* n + 1 - n
所以现在我们的目标是表明:
log(n)* n + 1 - n = O(log(n)* n)
显然,log(n)* n是O(log(n)* n),但是1-n
呢?
1-n = O(log(n)* n)?
1-n&lt; = k * log(n)* n,对于某些k?
设k = 1
1-n&lt; = log(n)* n?
向两侧添加n
1&lt; = n * log(n)+ n?的是
所以我们已经证明f(n)是O(n * log(n))。
现在我们使用log(n)* n在f(n)上有一个上界,让我们尝试使用log(n)* n得到f(n)的下界。
对于下限,我们使用 Big Omega 表示法。对于某些正常数k,Big Omega寻找函数g(n)* k <= f(n)。
k(n * log(n))&lt; = n * log(n)+ 1 - n?
设k = 1/10
n * log(n)/ 10&lt; = n * log(n)+ 1 - n?
(n * log(n) - 10n * log(n))/ 10&lt; = 1 - n?
-9n * log(n)/ 10&lt; = 1 - n?乘以10
-9n * log(n)&lt; = 10 - 10n?乘以-1(翻转不等式)
9n * log(n)&gt; = 10n - 10?除以9
n * log(n)&gt; = 10n / 9 - 10/9?除以n
log(n)&gt; = 10/9 - 10 / 9n?的是
显然,数量log(n)随着(10 / 9-10 / 9n)趋向于10/9而变大。实际上,对于n = 1,0> = 10/9 -10/9。 0> = 0。
所以现在我们已经展示了f(n) is Big-Omega(n*log(n))
。将此与f(n) is O(n*log(n))
的证明相结合,我们已经显示f(n) is Big-Theta(n*log(n))
! (感叹号是为了兴奋,而不是因子......)
g(n)= n * log(n),一组有效常数 k1 = 1/10 (下限)和 k2 = 1 (上限)。
答案 1 :(得分:1)
要证明big-O:外循环有floor(log2(n)) + 1 = O(log(n))
次迭代,内循环每次迭代O(n)
次,总共O(n * log(n))
。
要证明大欧米茄:在floor(log2(n/2)) + 1 = Omega(log(n))
期间,外循环有step <= n/2
次迭代。内部循环迭代n + 1 - step
次,对于这些外部迭代,这些次数超过n/2 = Omega(n)
,总共Omega(n * log(n))
。
同时,大O和大欧米茄证明了伟大的Theta。
答案 2 :(得分:1)
简化代码的表示,如下所示,我们可以将您的代码翻译成Sigma符号
for (step = 1; <= n; step = step * 2) {
for(i = n; i >= step; step = step - 1) {
}
}
像这样: