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
}
然而,这里我没有使用rowParser
和flatten
。我如何修改以前的代码以使用 rowParser
和 flatten
,而不将所有内容加载到内存中?像这样(注意:这段代码不起作用):
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
调用)
以下代码使用 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
}
然而,这里我没有使用rowParser
和flatten
。我如何修改以前的代码以使用 rowParser
和 flatten
,而不将所有内容加载到内存中?像这样(注意:这段代码不起作用):
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
调用)