Scala:如何在指定的索引范围内对数组进行排序?

时间:2013-07-10 11:07:44

标签: scala sorting stable-sort

我在代码中已经有一个比较函数“compr”来比较两个值。

我想要这样的事情:

Sorting.stableSort(arr[i,j] , compr)

其中arr [i,j]是数组中元素的范围。

4 个答案:

答案 0 :(得分:3)

将切片作为视图,对其进行排序和复制(或将切片作为工作缓冲区)。

scala> val vs = Array(3,2,8,5,4,9,1,10,6,7)
vs: Array[Int] = Array(3, 2, 8, 5, 4, 9, 1, 10, 6, 7)

scala> vs.view(2,5).toSeq.sorted.copyToArray(vs,2)

scala> vs
res31: Array[Int] = Array(3, 2, 4, 5, 8, 9, 1, 10, 6, 7)

在REPL之外,不需要额外的.toSeq

vs.view(2,5).sorted.copyToArray(vs,2)

答案 1 :(得分:1)

将数组拆分为三个部分,对中间部分进行排序然后将它们连接起来,而不是最有效的方式,但这是关心性能的FP =)

val sorted = 
  for {
    first       <- l.take(FROM)
    sortingPart <- l.slice(FROM, UNTIL)
    lastPart    <- l.takeRight(UNTIL)
  } yield (first ++ Sorter.sort(sortingPart) ++ lastPart)

答案 2 :(得分:1)

类似的东西:

def stableSort[T](x: Seq[T], i: Int, j: Int, comp: (T,T) => Boolean ):Seq[T] = {
    x.take(i) ++ x.slice(i,j).sortWith(comp) ++ x.drop(i+j-1)
}                                         

def comp: (Int,Int) => Boolean = { case (x1,x2) => x1 < x2 } 
val x = Array(1,9,5,6,3)                           
stableSort(x,1,4, comp)
// > res0: Seq[Int] = ArrayBuffer(1, 5, 6, 9, 3)

如果你的类实现了Ordering,那就不那么麻烦了。

答案 3 :(得分:0)

如果没有重新实现排序,这应该是你能得到的最好的。使用要排序的切片大小创建一个额外的数组。

def stableSort[K:reflect.ClassTag](xs:Array[K], from:Int, to:Int, comp:(K,K) => Boolean) : Unit = {
  val tmp = xs.slice(from,to)
  scala.util.Sorting.stableSort(tmp, comp)
  tmp.copyToArray(xs, from)
}