在Scala中调用基于模板类型的方法

在Scala中调用基于模板类型的方法,scala,generics,Scala,Generics,我试图找到一种从Scala调用JavaAPI的方法。基本上,有一个ContentValues对象,它有两个方法,如getAsString,getAsLong,每个方法都有各自不同的返回类型 我是否可以将ContentValues包装到另一个对象中,以便添加一个get[T]方法,根据T调用正确的getAsXXX方法 我尝试的(不起作用,抱怨含糊不清的隐式解决方案): 您可以使用与集合的CanBuildFrom特性相同的技术 我们首先创建一个Gettercase类 case class Getter

我试图找到一种从Scala调用JavaAPI的方法。基本上,有一个
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