4

为什么将:_*Seq 扩展为可变长度参数列表的 scala 在这种情况下不起作用?

以及如何优雅地解决它?

import java.sql.Connection
import scalikejdbc.ConnectionPool
import anorm.{SQL, SqlQuery, SqlRow, Row}

object AnormExample extends App {
  Class.forName("org.hsqldb.jdbc.JDBCDriver")
  ConnectionPool.singleton("jdbc:hsqldb:mem:hsqldb:WithAnorm", "", "")
  implicit val conn: Connection = ConnectionPool.borrow()

  // this works
  SQL("insert into emp (id, name) values ({id}, {name})").onParams(3, "name3").executeUpdate()  

  // this does not compile
  val row = Seq(4, "name4")
  SQL("insert into emp (id, name) values ({id}, {name})").onParams(row:_*).executeUpdate()  // david
}

错误:

scala: type mismatch;
 found   : Seq[Any]
 required: Seq[anorm.ParameterValue[?]]
  SQL("insert into emp (id, name) values ({id}, {name})").onParams(row:_*).executeUpdate()  // david

ps:

<dependency>
  <groupId>play</groupId>
  <artifactId>anorm_2.10</artifactId>
  <version>2.1.1</version>
</dependency>

<dependency>
  <groupId>com.github.seratch</groupId>
  <artifactId>scalikejdbc_2.10</artifactId>
  <version>1.5.1</version>
</dependency>

<dependency>
  <groupId>org.hsqldb</groupId>
  <artifactId>hsqldb</artifactId>
  <version>2.2.9</version>
</dependency>

任何优雅的解决方案?

更新

根据senia的回答,这解决了问题:

def toParameterValueSet(seq: Seq[Any]) = seq.map(v => v: anorm.ParameterValue[_])
val row = Seq(5, "name5")
SQL("insert into emp (id, name) values ({id}, {name})").onParams(toParameterValue(row):_*).executeUpdate()

有没有办法消除声明/使用 toParameterValueSet 的需要?

(例如,告诉 scala 在自动扩展上使用隐式转换:_*

更新

更紧凑一点:

implicit def toParameterValueSet(seq: Seq[Any]): Seq[anorm.ParameterValue[_]] = seq.map(v => v: anorm.ParameterValue[_])
val row = Seq(5, "name5")
SQL("insert into emp (id, name) values ({id}, {name})").onParams(row:_*).executeUpdate()

@Typesafe 团队:您可以在 acrom SQL 上添加一个参数来获取一系列值吗?(而不是必须使用:_*)(我同意使用命名参数是首选,但有时使用未命名参数仍然有用)

4

2 回答 2

3

你可以尝试更换

val row = Seq(4, "name4")

val row = Seq[anorm.ParameterValue[_]](4, "name4")

所有类型都有隐式转换,因此您可以使用row.map{ e => e: anorm.ParameterValue[_] }

于 2013-04-09T14:20:42.463 回答
2

只需指定type. 定义了<:一个类型绑定。

A <: B

说 A 必须受类型 B 的约束,或者任何 A 都是 B 的子类型。给它row:_*一个适当的type界限。我认为String界限有效。anorm.ParameterValue[String].

更具体地说,val row = Seq[anorm.ParameterValue[_]](4, "name4"). 这应该执行implicit转换。

于 2013-04-09T14:10:59.643 回答