Scala 为什么在使用val创建类参数时需要重写修饰符?
如果类参数声明为Scala 为什么在使用val创建类参数时需要重写修饰符?,scala,Scala,如果类参数声明为val而不是其他形式,为什么我需要override修饰符 scala> class A(val s:String){} defined class A scala> class B(val s:String) extends A(s) //doesnot work without override <console>:13: error: overriding value s in class A of type String; value s nee
val
而不是其他形式,为什么我需要override
修饰符
scala> class A(val s:String){}
defined class A
scala> class B(val s:String) extends A(s) //doesnot work without override
<console>:13: error: overriding value s in class A of type String;
value s needs `override' modifier
class B(val s:String) extends A(s)
^
scala> class A(s:String){}
defined class A
scala> class B(s:String) extends A(s) //override not required.
defined class B
scala>class-A(val-s:String){}
定义的A类
scala>class B(val s:String)扩展了A(s)//如果没有重写,就无法工作
:13:错误:重写字符串类型的A类中的值s;
值s需要“重写”修饰符
类B(val s:String)扩展了A(s)
^
scala>class A(s:String){}
定义的A类
scala>class B(s:String)扩展了A(s)//不需要重写。
定义的B类
在第二个示例中,不仅不需要覆盖
;这是不允许的,因为s
不会覆盖任何内容。如果构造函数参数不是val
,则表示它不构成类API的一部分,因此无需重写
在使用
val
时,如第一个示例中所示,除了构造函数参数s
之外,还隐式引入了名为s
的方法。因此,当您扩展A
时,Scala需要override
关键字来强制您承认您正在引入一个新的、不同的s
,该A
中的会隐藏起来;这是一件不寻常的事情,他们不希望你意外地做到这一点。类似的问题也得到了回答。显然编译器需要它。当查看带有“C:\Program Files\Java\jdk1.8.0\u 131\bin\javap”-p-C-constants-s-verbose B.class的“重写”示例的程序集时,确实创建了一个引用类A中字符串的额外字符串
{
public java.lang.String s();
descriptor: ()Ljava/lang/String;
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #12 // Method A.s:()Ljava/lang/String;
4: areturn
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this LB;
LineNumberTable:
line 8: 0
public B(java.lang.String);
descriptor: (Ljava/lang/String;)V
flags: ACC_PUBLIC
Code:
stack=2, locals=2, args_size=2
0: aload_0
1: aload_1
2: invokespecial #18 // Method A."<init>":(Ljava/lang/String;)V
5: return
LocalVariableTable:
Start Length Slot Name Signature
0 6 0 this LB;
0 6 1 s Ljava/lang/String;
LineNumberTable:
line 8: 0
}
{
public java.lang.String s();
描述符:()Ljava/lang/String;
旗帜:ACC_PUBLIC
代码:
堆栈=1,局部变量=1,参数大小=1
0:aload_0
1:invokespecial#12//方法A.s:()Ljava/lang/String;
4:轮到你了
LocalVariableTable:
起始长度插槽名称签名
0 5 0这磅;
LineNumberTable:
第8行:0
公共B(java.lang.String);
描述符:(Ljava/lang/String;)V
旗帜:ACC_PUBLIC
代码:
堆栈=2,局部变量=2,参数大小=2
0:aload_0
1:aload_1
2:invokespecial#18//方法A.“”:(Ljava/lang/String;)V
5:返回
LocalVariableTable:
起始长度插槽名称签名
0 6 0这磅;
0 6 1 s Ljava/lang/String;
LineNumberTable:
第8行:0
}
因此,如果我不提供像val
或var
这样的修饰符,Scala编译器将不会创建类成员。因此不需要重写。实际上,如果在构造函数外部使用它,它可能会创建一个私有成员,但这些成员不能被重写。