scala anorm foreach 没有将所有内容加载到内存中

scala anorm foreach without loading everything into memory

以下代码使用 scala anorm 打印所有行:

import anorm._

val query = s"select col_str, col_num from mytable"

val rowParser: RowParser[~[String, Int]] = SqlParser.str(1) ~ SqlParser.int(2)

def f(row: (String, Int)) {
  println(row._1 + "\t" + row._2)
}

val rows: List[(String, Int)] = SQL(query).as(rowParser.*).map(SqlParser.flatten)

rows.foreach(f)

但是它需要将所有数据加载到内存中。避免加载所有数据的解决方案是使用fold,如下:

SQL(query).fold(Unit, ColumnAliaser.empty) { (_, r: Row) =>
  println(r[String](1) + "\t" + r[Int](2))
  Unit
}

然而,这里我没有使用rowParserflatten。我如何修改以前的代码以使用 rowParserflatten,而不将所有内容加载到内存中?像这样(注意:这段代码不起作用):

SQL(query).as(rowParser.*).map(SqlParser.flatten).fold(Unit, ColumnAliaser.empty) { (_, row: (String, Int)) =>
  f(row)
  Unit
}

而且,更困难的是,必须创建一个隐式 forEach2 函数,以便我可以 运行 如下所示:

SQL(query).as(rowParser.*).map(SqlParser.flatten).forEach2(f)

我以前尝试过的一些代码:

def foreach[T, A, B](sqlQuery: SqlQuery, rowParser: RowParser[~[A, B]], f: (~[A, B]) => T) {
  val result: Either[List[Throwable], Unit.type] = sqlQuery.fold(Unit, ColumnAliaser.empty) { (_, row: Row) =>
    rowParser(row) match {
      case Success(r: ~[A, B]) =>
        f(r)
        Unit
      case Error(err) =>
        throw AnormException(err.toString)
    }
  }

  result.left.foreach { t: Seq[Throwable] =>
    t.foreach(_.printStackTrace)
    t.headOption.foreach { tt => throw tt}
  }
}

def f(row: ~[String, Int]) {
  println(row._1 + "\t" + row._2)
}

foreach(SQL(query), rowParser, f)

这行得通。但是,我需要将 def f(row: (String, Int)) 转换为 def f(row: ~[String, String])。如何在我的 f 函数中删除这个 ~?此外,此 foreach 函数需要一行两列。如何将其推广到 n 列?

你的 foreach 的改进是否足够好?或者您还需要更多吗?

object AnormForEachOps {

  import anorm._
  import java.sql.Connection

  implicit class ForEachOps(val query: SqlQuery) extends AnyVal {
    def foreach[T1, T2, R](rowParser: RowParser[~[T1, T2]], f: R => Unit)(implicit connection: Connection, fl: TupleFlattener[(T1 ~ T2) => R]): Unit = {
      query.fold(Unit, ColumnAliaser.empty) { (_, r: Row) =>
        rowParser(r).map(SqlParser.flatten) match {
          case Success(t) =>
            f(t)
            Unit
          case Error(err) =>
            throw AnormException(err.toString)
        }
        Unit
      }
    }
  }

}

所以你的例子会变成这样:

import AnormForEachOps._
val query = s"select col_str, col_num from mytable"
val rowParser = SqlParser.str(1) ~ SqlParser.int(2)
SQL(query).foreach(rowParser, (r: (String, Int)) => {
    Logger.info(r._1 + "\t" + r._2)
})

此修改中的主要技巧是隐式 TupleFlattener 参数,它执行从 ~ 到普通元组的转换(通过 SqlParser.flatten 调用)