在Scala中调用基于模板类型的方法
我试图找到一种从Scala调用JavaAPI的方法。基本上,有一个在Scala中调用基于模板类型的方法,scala,generics,Scala,Generics,我试图找到一种从Scala调用JavaAPI的方法。基本上,有一个ContentValues对象,它有两个方法,如getAsString,getAsLong,每个方法都有各自不同的返回类型 我是否可以将ContentValues包装到另一个对象中,以便添加一个get[T]方法,根据T调用正确的getAsXXX方法 我尝试的(不起作用,抱怨含糊不清的隐式解决方案): 您可以使用与集合的CanBuildFrom特性相同的技术 我们首先创建一个Gettercase类 case class Getter
ContentValues
对象,它有两个方法,如getAsString
,getAsLong
,每个方法都有各自不同的返回类型
我是否可以将ContentValues
包装到另一个对象中,以便添加一个get[T]
方法,根据T调用正确的getAsXXX
方法
我尝试的(不起作用,抱怨含糊不清的隐式解决方案):
您可以使用与集合的
CanBuildFrom
特性相同的技术
我们首先创建一个Getter
case类
case class Getter[T](getter: (ContentValues, String) => T) {
def getOpt(contentValues: ContentValues, key: String): Option[T] =
if (contentValues containsKey key) Some(getter(contentValues, key))
else None
}
这允许我们创建一个具有所需方法的ContentValues
包装器
implicit class ContentValuesWrapper(val c: ContentValues) extends AnyVal {
def getAsOpt[T](key: String)(implicit getter: Getter[T]) =
getter.getOpt(c, key)
}
现在,为了调用ContentValues
上的getAsOpt
方法,我们需要为正确的类型提供Getter
的implicit
实例
object Getter {
implicit val intGetter = Getter(_ getAsInteger _)
implicit val longGetter = Getter(_ getAsLong _)
implicit val stringGetter = Getter(_ getAsString _)
}
现在,您可以在ContentValues
实例上使用getAsOpt
方法
// fake version of ContentValues
val c =
new ContentValues {
val m = Map("a" -> "1", "b" -> "2", "c" -> "3")
def getAsInteger(k: String): Int = getAsString(k).toInt
def getAsLong(k: String): Long = getAsString(k).toLong
def getAsString(k: String): String = m(k)
def containsKey(k: String): Boolean = m contains k
}
c.getAsOpt[Int]("a") //Option[Int] = Some(1)
c.getAsOpt[Long]("b") //Option[Long] = Some(2)
c.getAsOpt[String]("c") //Option[String] = Some(3)
c.getAsOpt[Int]("d") //Option[Int] = None
c.getAsOpt[Long]("e") //Option[Long] = None
c.getAsOpt[String]("f") //Option[String] = None
谢谢!我不太清楚为什么我的方法行不通,但我还是不说了;)您有三个不可区分的隐式,它们添加了一个
getTyped
方法。编译器没有足够的信息从一个名为c.getTyped(“x”)
Ah的方法中选择正确的一个,是的,所以参数化getAsOpt
做了这个把戏,并告诉编译器足够的信息来区分转换?我想这是有道理的+1对于Getter(\uGetAsInteger)
语法-你在哪里找到的?@sourcedelica过去几个月我一直在摆弄一个框架,最终用户的语法对我来说非常重要。这让我认识到,几乎所有重复的代码都可以用Scala消除。我觉得解决这些难题很有意思。
// fake version of ContentValues
val c =
new ContentValues {
val m = Map("a" -> "1", "b" -> "2", "c" -> "3")
def getAsInteger(k: String): Int = getAsString(k).toInt
def getAsLong(k: String): Long = getAsString(k).toLong
def getAsString(k: String): String = m(k)
def containsKey(k: String): Boolean = m contains k
}
c.getAsOpt[Int]("a") //Option[Int] = Some(1)
c.getAsOpt[Long]("b") //Option[Long] = Some(2)
c.getAsOpt[String]("c") //Option[String] = Some(3)
c.getAsOpt[Int]("d") //Option[Int] = None
c.getAsOpt[Long]("e") //Option[Long] = None
c.getAsOpt[String]("f") //Option[String] = None