最近用slick作为数据库访问框架,有些功能不好实现,用plain SQL,sql是根据查询条件生成的,所以参数也不是固定的个数。但StaticQuery[T, Entity]里面的T,默认只支持数据库支持的类型比如Int,Double等,以及对应的Optional,也可以支持Tuple1--Tuple22,只要里面的类型是数据库支持的类型。但因为我要传递的参数在编译期没法知道有多少个,所以只能用List[Int]来传递,结果报错说找不到对应的隐式转化。
[error] D:\xx.scala:49: could not find implicit value for parameter pconv1: scala.slick.jdbc.SetParameter[List[Int]] [error] val query = Q[List[Int], Entity] + sql [error] ^ [error] one error found [error] (compile:compile) Compilation failed
然后就想看看StaticQuery这个类具体做了什么,点进去看源码,发现它需要一个implicit的参数
object StaticQuery { ............. def apply[P, R](implicit pconv1: SetParameter[P], rconv: GetResult[R]) = query[P,R]("") ............ }
trait SetParameter[-T] extends ((T, PositionedParameters) => Unit) { self => def applied(value: T): SetParameter[Unit] = new SetParameter[Unit] { def apply(u: Unit, pp: PositionedParameters) { self.apply(value, pp) } } }
implicit object SetInt extends SetParameter[Int] { def apply(v: Int, pp: PositionedParameters) { pp.setInt(v) } }
implicit object SetListInt extends SetParameter[List[Int]] { def apply(vList: List[Int], pp: PositionedParameters) { for (v <- vList) pp.setInt(v) } }
总结
以后遇到问题,首先想想是否可以转化为以后的实现方式(List[Int] -> TupleX[Int]);如果不行,Google、*等查查有没有其他解决方法;最后不行,就只能自己看source code去掌握原理,自己实现了。