我如何在Scala中重新实现Tie :: File?

Sen*_*ess 6 perl scala

我真的很喜欢Tie :: File,它允许你将tie数组转换为文件的行.您可以以任何方式修改数组,当您完成它时,您untie可以相应地修改该文件的内容.

我想在Scala中重新实现这种行为,这是我到目前为止所做的:

class TiedBuffer(val file:File) extends ArrayBuffer[String] {

  tieFile

  def untie = {
      val writer = new PrintStream(new FileOutputStream(file))
      this.foreach(e => writer.println(e))
      writer.close
      this
  }

  private def tieFile = this ++= scala.io.Source.fromFile(file).getLines()
}
Run Code Online (Sandbox Code Playgroud)

但是,ArrayBuffer上定义的"运算符"返回各种类,不同于我自己的类,例如:

println((new TiedBuffer(somefile) +: "line0").getClass)
Run Code Online (Sandbox Code Playgroud)

给了我一个immutable.Vector.我可以将类限制为一组非常小的预定义方法,但我认为如果我可以提供所有这些方法(foreach/map/...)会很好.

我应该继承什么,或者我应该如何解决这个问题,以便我有一个类似流体的数组接口,它允许我修改文件的内容?

BOUNTY:为了赢得赏金,你能展示一个可以CanBuildFrom用来完成这项任务的工作实例吗?

Mar*_*ela 4

以冒号结尾的方法是右关联的,因此在您的示例中,您使用as 参数+:进行调用。如果你想测试你可以这样做:StringTiedBuffer+:ArrayBuffer

println((new TiedBuffer(somefile).+:("line0")).getClass)
Run Code Online (Sandbox Code Playgroud)

或者

println(("line0" +: new TiedBuffer(somefile)).getClass)
Run Code Online (Sandbox Code Playgroud)

编辑

我错过了你问题中的要点,请参阅约翰对返回对象而不是 的回答TiedBufferArrayBuffer

编辑2

这是一个带有 的示例CanBuildFrom。不过,您必须tie手动调用,以防止每次构建器创建新实例时都绑定文件TiedBuffer。还有很大的改进空间,例如++不起作用,但它应该可以帮助您入门。

import collection.generic.CanBuildFrom
import collection.mutable._
import java.io.{PrintStream, FileOutputStream, File}

class TiedBuffer(val file: File) extends ArrayBuffer[String]
                                 with BufferLike[String, TiedBuffer]
                                 with IndexedSeqOptimized[String, TiedBuffer] {

  def tie = {
    clear
    this ++= scala.io.Source.fromFile(file).getLines()
  }

  def untie = {
    val writer = new PrintStream(new FileOutputStream(file))
    this.foreach(e => writer.println(e))
    writer.close
    this
  }

  override def newBuilder: Builder[String, TiedBuffer] =
    new ArrayBuffer mapResult {
      x: Seq[String] => (new TiedBuffer(file) ++= x)
    }
}

object TiedBuffer {
  implicit def canBuildFrom: CanBuildFrom[TiedBuffer, String, TiedBuffer] =
    new CanBuildFrom[TiedBuffer, String, TiedBuffer] {
      def apply(): Builder[String, TiedBuffer] =
        throw new RuntimeException("Cannot create a new TiedBuffer from scratch")

      def apply(from: TiedBuffer): Builder[String, TiedBuffer] = from.newBuilder
    }
}
Run Code Online (Sandbox Code Playgroud)