24

我正在尝试使用一些采用隐式参数的方法来定义一个类:

object Greetings {
  def say(name: String)(implicit greetings: String): String = greetings + " " +name 
}

我从另一个班级使用这个班级

implicit val greetings = "hello"                //> greetings  : java.lang.String = hello
Greetings.say("loic")                           //> res0: String = hello loic
Greetings.say("loic")("hi")                     //> res1: String = hi loic

我的问题是它只有在我在 Greetings 对象之外定义隐式 val 时才有效。我希望能够提供带有隐式参数的方法,并在我的类中使用默认值,以便更轻松地使用我的 API(如 Scala 集合 API)。

所以我想这样做,但它不起作用(未找到隐式值):

object Greetings {
  implicit val greetings = "hello"    
  def say(name: String)(implicit greetings: String): String = greetings + " " +name 
}

进而

Greetings.say("loic")                         
Greetings.say("loic")("hi") 

我知道我可以定义一个默认值,(implicit greetings: String = "hello")但我想在类级别进行,以避免在有很多方法时重复。

我想我错过了一些东西,因为我看到它CanBuildFrom是在List类中定义的,例如。

4

2 回答 2

26

String在隐式中使用这样的通用类型是个坏主意。主要原因是隐式查找完全基于类型,那么如果其他人定义了另一个 String 类型的隐式值怎么办?你最终可能会发生冲突。因此,您应该为自己的目的定义自己的特定类型(字符串的简单包装器)。

另一个原因是,在查找隐式值时,编译器将(在其他地方)查找隐式值类型的伴随对象(如果有的话)。您可以轻松地看到它有多么有用,因为伴随对象是放置默认隐式值的自然位置(如您的情况)。但是,如果隐式值是您不拥有的类型(例如String),您就不能为其编写伴随对象,而使用您自己的包装器类型则没有问题。

好的,废话不多说,这是你可以做到的:

case class Greetings( value: String ) {
  override def toString = value
}
object Greetings {
  // this implicit is just so that we don't have to manually wrap 
  // the string when explicitly passing a Greetings instance
  implicit def stringToGreetings( value: String ) = Greetings( value ) 

  // default implicit Greetings value
  implicit val greetings: Greetings ="hello"

  def say(name: String)(implicit greetings: Greetings): String = greetings + " " +name 
}
Greetings.say("loic")                         
Greetings.say("loic")("hi") 
于 2012-10-07T12:11:56.767 回答
12

我找到了一种解决方法:

class Greetings(implicit val greetings: String = "hello") {
    def say(name: String): String = greetings + " " + name 
}

像这样,我可以有一个默认值并在需要时覆盖它:

new Greetings().say("loic")                     //> res0: String = hello loic

implicit val greetings = "hi"                   //> greetings  : java.lang.String = hi
new Greetings().say("loic")                     //> res1: String = hi loic

new Greetings()("coucou").say("loic")           //> res2: String = coucou loic

注意:new Greetings()("coucou")正在工作,而不是,因为这里new Greetings("coucou")解释的语法奇怪。

于 2012-10-07T08:57:31.077 回答