普通数组上的 Haskell 快速排序 - 可能吗?

Posted

技术标签:

【中文标题】普通数组上的 Haskell 快速排序 - 可能吗?【英文标题】:Haskell quicksort on plain arrays - possible? 【发布时间】:2021-01-05 21:17:44 【问题描述】:

在自学haskell 时,我正在尝试实现快速排序。为方便起见,我将自己限制在 ghc 中包含的包中,这就是我使用 Array 而不是 Vector 的原因:

import Data.Array

type ArrayT = Array Int

quickSort :: Ord t => ArrayT t -> ArrayT t
quickSort a = rQuickSort (bounds a) a

rQuickSort :: Ord t => (Int, Int) -> ArrayT t -> ArrayT t
rQuickSort (beg,end) arr | beg >= end = arr
rQuickSort (beg,end) arr = 
  rQuickSort (beg, pivIndex) . rQuickSort (pivIndex+1, end) $ parted 
  where
    (pivoted, pivValue) = med3pivot (beg, end) arr
    (parted , pivIndex) = partition (beg+1, end-1) pivoted pivValue

med3pivot :: Ord t => (Int, Int) -> ArrayT t -> (ArrayT t, t)
med3pivot (beg, end) = 
  let mid = (beg + end) `div` 2 
    in (\arr -> (arr,arr!mid))
    . (\arr -> if arr!end < arr!mid then swap end mid arr else arr)
    . (\arr -> if arr!mid < arr!beg then swap mid beg arr else arr)
    . (\arr -> if arr!end < arr!beg then swap end beg arr else arr)
    
partition :: Ord t => (Int, Int) -> ArrayT t -> t -> (ArrayT t, Int)
partition (beg, end) arr piv =
  let fw = findForward  (piv <) (beg,end) arr
      bw = findBackward (piv >) (beg,end) arr
    in if fw >= bw then (arr, bw) 
                   else partition (fw+1, bw-1) (swap fw bw arr) piv

findForward :: (t -> Bool) -> (Int, Int) -> ArrayT t -> Int
findForward _ (b,e) _ | b > e = b
findForward p (b,e) a = if p (a!b) then b else findForward p (b+1,e) a

findBackward :: (t -> Bool) -> (Int, Int) -> ArrayT t -> Int
findBackward _ (b,e) _ | b > e = e
findBackward p (b,e) a = if p (a!e) then e else findBackward p (b,e-1) a 

swap :: Int -> Int -> ArrayT t -> ArrayT t
swap i j arr = arr // [(i, arr!j), (j, arr!i)]

mkArray :: [a] -> Array Int a
mkArray xs = listArray (0, length xs - 1) xs

我显然弄错了。排序 10000 个短字节字符串数组的运行时间约为 8 秒(暂时省略驱动代码,因为这已经是一堆了)。 我想我没有正确理解一般的懒惰,以及数组副本发生在哪里。任何指针表示赞赏。

(我找到了这个答案https://***.com/a/5269180/11061421,它使用了 Array.ST,但我仍然想知道我的代码有什么问题。甚至可以像我正在尝试的那样对普通数组进行就地排序吗? )

【问题讨论】:

除了 Jon Purdy 所说的,如果您不使用随机枢轴或使用中位数或类似方法选择枢轴,这真的不是 QuickSort。 【参考方案1】:

Array 是一种不可变的数据类型,因此每次交换都会构造一个新数组“脊椎”,尽管元素(即 t 类型的值)是共享的。

您通常应该使用Array 来计算诸如记忆表之类的东西,例如从其他语言翻译动态编程算法时,而不是像这样的顺序算法。当您使用相互引用(无循环)的 thunk 填充元素并从索引中读取以计算和缓存该索引的元素时,数组效果最佳。

获得实际就地排序的唯一方法是在STIO 中使用可变数组类型。但是,我认为您也可以通过构建一系列条件交换(即排序网络)并仅将其应用于输入数组 once 来更有效地进行这种不可变数组排序以产生输出数组。

【讨论】:

以上是关于普通数组上的 Haskell 快速排序 - 可能吗?的主要内容,如果未能解决你的问题,请参考以下文章

为啥极简主义,例如 Haskell 快速排序不是“真正的”快速排序?

500,000 个已排序整数数组上的 C++ 快速排序算法中的 Seg 错误

Haskell学习-常见排序算法汇总

快速排序是一种分而治之的方法吗? [关闭]

Haskell:当不需要日志时,让 Writer 和普通代码一样高效

leedcode算法刷题之利用普通排序算法思路