我正在尝试实现我刚刚在课堂上学到的 rselect 算法。但是,似乎无法弄清楚我在实施中哪里出错了。这是我的代码。*编辑 * :我尝试使用大卫回答中提供的信息,但我的代码仍然很奇怪。这是修改后的代码:
def rselect(seq,length,i):# i is the i'th order statistic.
if len(seq)<=1:return seq
lo,pi,hi,loc_pi= random_partition(seq
if loc_pi==i:return pi
if loc_pi>i:return rselect(lo,loc_pi-1,i)
elif loc_pi<i:return rselect(hi,length-loc_pi,i-loc_pi)#
from random import choice
def random_partition(seq):
pi =choice(seq)
#print 'pi',pi
loc_pi=seq.index(pi)
print 'Location',loc_pi
lo=[x for x in seq if x<=pi]
hi=[x for x in seq if x>pi]
return lo,pi,hi,len(lo)+1 #--A
def test_rselect(seq,i):
print 'Sequence',seq
l=len(seq)
print 'Statistic', rselect(seq,l,i)
然而,输出在不同的时间是不同的,有时甚至是正确的!我是算法和 python 的菜鸟,任何关于我哪里出错的帮助将不胜感激。编辑:每次运行代码时,我都会为第 i 个顺序统计数据获取不同的值,这是我的问题 例如,每次运行代码如下所示
Revised Output:
/py-scripts$ python quicksort.py
Sequence [54, -1, 1000, 565, 64, 2, 5]
Statistic Location 1
-1
@ubuntu:~/py-scripts$ python quicksort.py
Sequence [54, -1, 1000, 565, 64, 2, 5]
Statistic Location 5
Location 1
Location 0
-1
预期输出:我期待在这里找到第 i 个顺序统计。
因此
test_rselect([54,-1,1000,565,64,2,5],2)
应该一直5
作为统计返回。
任何帮助我在这个实现中出错的地方都会有帮助..谢谢!
编辑 2:通过尝试分析算法,我相信错误在于我如何在标记为 A 的行中返回枢轴位置(loc_pi)。考虑到上述程序的以下事件序列。
test_rselect( [ 55, 900, -1,10, 545, 250], 3) // call to input array
calls rselect ([ 55, 900, -1,10, 545, 250],6,3)
1st call to random_partition:
pi=545 and loc_pi=4
lo=[55,-1,10,250,545]
hi=[900]
return to rselect function (lo,545,hi,6)
here loc_pi>i: so rselect(lo,5,3)// and discard the hi part
2nd recursive call to rselect:
2nd recursive call to random_partition:
call random_partition on (lo) // as 'hi' is discarded
pi=55 loc_pi=0
lo=[-1,10,55]
hi=[250,545]
return to rselect(lo,55,hi,4)
here loc_pi>i: rselect(lo,3,3)// The pivot element is lost already as it is in 'hi' here!!
关于如何处理返回枢轴元素的位置以获得正确的 o/p 的任何帮助都会有所帮助。设置赏金,以获得清楚地解释我做错了什么以及如何纠正它的答案(欢迎伟大的提示,因为我期待学习:))。期待伟大的答案!