Scala:在不创建对象的情况下获取默认值

时间:2015-12-08 16:08:44

标签: scala default-value

假设构造函数中有一个具有默认值的类:

class Adapter(url: String = "127.0.0.1", port: Int = 8080)

是否可以在运行时获取这些参数的默认值而无需创建此对象的实例?

请不要询问用例,这是一个关于语言本身的问题。

2 个答案:

答案 0 :(得分:6)

我假设您最感兴趣的是如何在内部实现默认方法。以下是代码的javap反汇编(使用非常有用的:javap <classname> scala控制台命令:

scala> class Adapter(url: String = "127.0.0.1", port: Int = 8080)
defined class Adapter

scala> :javap -c Adapter$ // Adapter$ because we want the companion object
Compiled from "<console>"
public class Adapter$ {

  ... 

  public java.lang.String $lessinit$greater$default$1();
    Code:
       0: ldc           #16                 // String 127.0.0.1
       2: areturn       

  public int $lessinit$greater$default$2();
    Code:
       0: sipush        8080
       3: ireturn       

  ...

如您所见,创建默认值的方法在伴随对象中编码为特别命名的方法。因此,要在不重构示例的情况下调用这些方法,您必须使用反射:

scala> Adapter.getClass.getMethod("$lessinit$greater$default$1").invoke(Adapter) 
res8: Object = 127.0.0.1

scala> Adapter.getClass.getMethod("$lessinit$greater$default$2").invoke(Adapter) 
res9: Object = 8080

显然,在生产代码中执行此操作可能不是一个好主意,因为名称修改细节等可能会发生变化。在这种情况下,只需使用@Sparko的建议。

关于名称修改方案的说明:该方案是methodName + "$default$" + parameterNumber。在上面的例子中要装饰的方法是构造函数,它是“&lt; init&gt;”。所以你最终得到了方法名称的“$ lessinit $ greater”和计算第一个参数的默认值的方法名称的“$ lessinit $ greater $ default $ 0”。

答案 1 :(得分:2)

将它们移动到伴侣对象中?

case class Adapter(url: String = Adapter.defaultURL, port: Int = Adapter.defaultPort)

object Adapter {
    val defaultURL = "127.0.0.1"
    val defaultPort = 8080
}

如果只能通过private

访问默认值,则可以使用case class确定其他默认值的范围