快速排序返回错误的顺序
Quicksort returning wrong order
我正在 Python 中使用递归实现快速排序,而无需创建新变量来保存分区数组的左右部分。
我在 运行 执行递归步骤时得到了错误的值。这是我到目前为止所做的:
def swap(a,i,j):
tmp = a[i]
a[i] = a[j]
a[j] = tmp
def pivot(a, lo, hi):
mid = (lo + hi) // 2
# sort lo, mid, hi:
if a[mid] < a[lo]:
swap(a, lo, mid)
if a[hi] < a[lo]:
swap(a, lo, hi)
if a[hi] < a[mid]:
swap(a, mid, hi)
def partition(a, lo, hi):
# place the pivot out of the way, in position hi -1
mid = (lo + hi)//2
swap(a, mid, hi - 1)
i = lo
j = hi - 1
pivot = a[j]
while True:
while True:
i += 1
if a[i] >= pivot: break
while True:
j -= 1
if a[j] <= pivot: break
if i >= j: break
swap(a, i, j)
swap(a, i, hi - 1)
return i
假设上面的模板代码是“正确的”。我必须使用上面的 pivot 和 partition 实现来实现快速排序。这就是我所做的:
def quicksort(a):
_sort(a, 0, len(a)-1)
def _sort(a, left, right):
if(left < right):
pivot(a, left, right)
piv = partition(a, left, right)
_sort(a, left, piv-1)
_sort(a, piv+1, right)
当我用列表调用快速排序时:
x = [98, 33, 11, 5, 1, 10, 11, 12, 14, 33, 55, 66, 556, 88]
quicksort(x)
print(x)
>>> [1, 5, 10, 11, 11, 12, 14, 33, 33, 55, 66, 88, 556, 98]
可以看到98错位了。如果我 运行 这样:
x = [9, 7, 5, 11, 12, 2, 14, 3, 10, 6, 55, 66, 888, 33, 556, 10]
quicksort(x)
print(x)
>>> [2, 3, 5, 6, 7, 9, 10, 10, 11, 12, 14, 33, 55, 66, 556, 888]
因此,对于上述情况,它是正确的。但在其他较小的情况下它失败了:
x = [98, 33, 11, 556, 88]
quicksort(x)
print(x)
>>> [33, 11, 88, 556, 98]
谁能帮我找出错误?谢谢:-)
我看到两个错误,第一个在您的 partition()
函数中。
尝试用 swap(a, i, hi)
替换 swap(a, i, hi - 1)
第二个在 _sort()
。最后一次调用应该是 _sort(a, piv, right)
正确的代码是:
def partition(a, lo, hi):
# place the pivot out of the way, in position hi -1
mid = (lo + hi)//2
swap(a, mid, hi - 1)
i = lo
j = hi - 1
pivot = a[j]
while True:
while True:
i += 1
if a[i] >= pivot: break
while True:
j -= 1
if a[j] <= pivot: break
if i >= j: break
swap(a, i, j)
swap(a, i, hi)
return i
和
def _sort(a, left, right):
if(left < right):
pivot(a, left, right)
piv = partition(a, left, right)
_sort(a, left, piv-1)
_sort(a, piv, right)
测试:
x = [98, 33, 11, 556, 88]
quicksort(x)
print(x)
[11, 33, 88, 98, 556]
更多测试:
import random
for n in range(10):
x = [random.randint(1,999) for i in range(random.randint(4,10))]
quicksort(x)
print(x)
[104, 226, 721, 769]
[131, 453, 590, 730, 752, 834]
[132, 156, 191, 277, 541, 599, 666, 909, 919]
[114, 210, 280, 919, 968, 978]
[127, 212, 381, 458, 585, 594, 685, 809, 935]
[73, 90, 189, 591, 599, 686, 806, 829, 831, 906]
[89, 115, 208, 601, 774, 813, 842, 981]
[159, 177, 203, 231, 621, 759, 950]
[347, 348, 417, 476, 850, 902]
[8, 50, 51, 483, 499, 696, 842]
我正在 Python 中使用递归实现快速排序,而无需创建新变量来保存分区数组的左右部分。
我在 运行 执行递归步骤时得到了错误的值。这是我到目前为止所做的:
def swap(a,i,j):
tmp = a[i]
a[i] = a[j]
a[j] = tmp
def pivot(a, lo, hi):
mid = (lo + hi) // 2
# sort lo, mid, hi:
if a[mid] < a[lo]:
swap(a, lo, mid)
if a[hi] < a[lo]:
swap(a, lo, hi)
if a[hi] < a[mid]:
swap(a, mid, hi)
def partition(a, lo, hi):
# place the pivot out of the way, in position hi -1
mid = (lo + hi)//2
swap(a, mid, hi - 1)
i = lo
j = hi - 1
pivot = a[j]
while True:
while True:
i += 1
if a[i] >= pivot: break
while True:
j -= 1
if a[j] <= pivot: break
if i >= j: break
swap(a, i, j)
swap(a, i, hi - 1)
return i
假设上面的模板代码是“正确的”。我必须使用上面的 pivot 和 partition 实现来实现快速排序。这就是我所做的:
def quicksort(a):
_sort(a, 0, len(a)-1)
def _sort(a, left, right):
if(left < right):
pivot(a, left, right)
piv = partition(a, left, right)
_sort(a, left, piv-1)
_sort(a, piv+1, right)
当我用列表调用快速排序时:
x = [98, 33, 11, 5, 1, 10, 11, 12, 14, 33, 55, 66, 556, 88]
quicksort(x)
print(x)
>>> [1, 5, 10, 11, 11, 12, 14, 33, 33, 55, 66, 88, 556, 98]
可以看到98错位了。如果我 运行 这样:
x = [9, 7, 5, 11, 12, 2, 14, 3, 10, 6, 55, 66, 888, 33, 556, 10]
quicksort(x)
print(x)
>>> [2, 3, 5, 6, 7, 9, 10, 10, 11, 12, 14, 33, 55, 66, 556, 888]
因此,对于上述情况,它是正确的。但在其他较小的情况下它失败了:
x = [98, 33, 11, 556, 88]
quicksort(x)
print(x)
>>> [33, 11, 88, 556, 98]
谁能帮我找出错误?谢谢:-)
我看到两个错误,第一个在您的 partition()
函数中。
尝试用 swap(a, i, hi)
swap(a, i, hi - 1)
第二个在 _sort()
。最后一次调用应该是 _sort(a, piv, right)
正确的代码是:
def partition(a, lo, hi):
# place the pivot out of the way, in position hi -1
mid = (lo + hi)//2
swap(a, mid, hi - 1)
i = lo
j = hi - 1
pivot = a[j]
while True:
while True:
i += 1
if a[i] >= pivot: break
while True:
j -= 1
if a[j] <= pivot: break
if i >= j: break
swap(a, i, j)
swap(a, i, hi)
return i
和
def _sort(a, left, right):
if(left < right):
pivot(a, left, right)
piv = partition(a, left, right)
_sort(a, left, piv-1)
_sort(a, piv, right)
测试:
x = [98, 33, 11, 556, 88]
quicksort(x)
print(x)
[11, 33, 88, 98, 556]
更多测试:
import random
for n in range(10):
x = [random.randint(1,999) for i in range(random.randint(4,10))]
quicksort(x)
print(x)
[104, 226, 721, 769]
[131, 453, 590, 730, 752, 834]
[132, 156, 191, 277, 541, 599, 666, 909, 919]
[114, 210, 280, 919, 968, 978]
[127, 212, 381, 458, 585, 594, 685, 809, 935]
[73, 90, 189, 591, 599, 686, 806, 829, 831, 906]
[89, 115, 208, 601, 774, 813, 842, 981]
[159, 177, 203, 231, 621, 759, 950]
[347, 348, 417, 476, 850, 902]
[8, 50, 51, 483, 499, 696, 842]