假设构造函数中有一个具有默认值的类:
class Adapter(url: String = "127.0.0.1", port: Int = 8080)
是否可以在运行时获取这些参数的默认值而无需创建此对象的实例?
请不要询问用例,这是一个关于语言本身的问题。
答案 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
确定其他默认值的范围