首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >就地混洗ArrayBuffer的一部分

就地混洗ArrayBuffer的一部分
EN

Stack Overflow用户
提问于 2012-03-06 03:07:15
回答 2查看 1.1K关注 0票数 5

我需要混洗ArrayBuffer的一部分,最好是就地,这样就不需要拷贝了。例如,如果一个ArrayBuffer有10个元素,而我想要将元素3-7打乱:

代码语言:javascript
运行
复制
// Unshuffled ArrayBuffer of ints numbered 0-9
0, 1, 2, 3, 4, 5, 6, 7, 8, 9

// Region I want to shuffle is between the pipe symbols (3-7)
0, 1, 2 | 3, 4, 5, 6, 7 | 8, 9

// Example of how it might look after shuffling
0, 1, 2 | 6, 3, 5, 7, 4 | 8, 9

// Leaving us with a partially shuffled ArrayBuffer
0, 1, 2, 6, 3, 5, 7, 4, 8, 9

我已经编写了如下所示的代码,但它需要复制和迭代循环几次。似乎应该有一种更有效的方法来做这件事。

代码语言:javascript
运行
复制
def shufflePart(startIndex: Int, endIndex: Int) {

  val part: ArrayBuffer[Int] = ArrayBuffer[Int]()

  for (i <- startIndex to endIndex ) {
    part += this.children(i)
  }

  // Shuffle the part of the array we copied
  val shuffled = this.random.shuffle(part)
  var count: Int = 0

  // Overwrite the part of the array we chose with the shuffled version of it
  for (i <- startIndex to endIndex ) {
    this.children(i) = shuffled(count)
    count += 1
  }
}

我找不到任何关于与谷歌部分洗牌ArrayBuffer的内容。我假设我将不得不写我自己的方法,但在这样做的时候,我想防止复制。

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2012-03-06 03:31:13

我不完全确定为什么它必须在适当的地方-您可能需要考虑这一点。但是,假设这是正确的做法,这是可以做到的:

代码语言:javascript
运行
复制
import scala.collection.mutable.ArrayBuffer

implicit def array2Shufflable[T](a: ArrayBuffer[T]) = new {
  def shufflePart(start: Int, end: Int) = {
    val seq = (start.max(0) until end.min(a.size - 1)).toSeq
    seq.zip(scala.util.Random.shuffle(seq)) foreach { t =>
      val x = a(t._1)
      a.update(t._1, a(t._2))
      a(t._2) = x
    }
    a
  }
}

您可以像这样使用它:

代码语言:javascript
运行
复制
val a = ArrayBuffer(1,2,3,4,5,6,7,8,9)
println(a)
println(a.shufflePart(2, 7))  

编辑:如果你愿意支付中间序列的额外成本,从算法上讲,这是更合理的:

代码语言:javascript
运行
复制
  def shufflePart(start: Int, end: Int) = {
    val seq = (start.max(0) until end.min(a.size - 1)).toSeq
    seq.zip(scala.util.Random.shuffle(seq) map { i =>
      a(i)
    }) foreach { t =>
      a.update(t._1, t._2)
    }
    a
  }
}
票数 3
EN

Stack Overflow用户

发布于 2012-03-06 09:41:59

因为ResizableArray有一个受保护的成员array,所以如果可以使用ArrayBuffer的子类型,就可以直接访问底层数组

代码语言:javascript
运行
复制
import java.util.Collections
import java.util.Arrays
import collection.mutable.ArrayBuffer


val xs = new ArrayBuffer[Int]() {
  def shuffle(a: Int, b: Int) {
    Collections.shuffle(Arrays.asList(array: _*).subList(a, b))
  }
}

xs ++= (0 to 9)    // xs = ArrayBuffer(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)
xs.shuffle(3, 8)   // xs = ArrayBuffer(0, 1, 2, 4, 6, 5, 7, 3, 8, 9)

备注:

  • java.util.List#subList的上限是独占的
  • 它的效率相当高,因为Arrays#asList不会创建新的元素集:它是由数组本身支持的(因此没有添加或删除方法)
  • 如果使用的是实数,您可能需要对aArrays#asList添加边界检查
票数 5
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/9572408

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档