了解Scala中的Case类和Traits

spa*_*rkr 11 scala

我有一个如下定义的简单特征:

trait MyTrait {

  def myStringVal: String

}
Run Code Online (Sandbox Code Playgroud)

我的case类实现了这个特性如下:

case class MyCaseClass(myStringVal: String) extends MyTrait {
  ...
  ...
}
Run Code Online (Sandbox Code Playgroud)

来自Java世界,我发现有点难以理解MyCaseClass实际上只是通过定义MyCaseClass的参数来实现它.我知道你的字节代码实际上会编写getter和setter.但如果没有var或val,这怎么可能呢?

我的理解是,如果没有var或val,则不会生成getter或setter方法.在这种情况下,上面的案例类MyCaseClass如何实现myStringVal方法?

有时太多的Scala魔法难以理解,特别是遗留代码.

Flo*_*her 21

您可能希望查看此博客文章,其中涵盖了具体类型以及它们如此有用的原因.

在您的示例中,MyTrait除了能够像java接口一样运行之外,该特征没有用处.请注意,scala中的默认可见性是公共的.默认情况下,case类参数是不可变的,因此在您的示例val中,编译器会自动推断该myStringVal参数.

案例课有什么神奇之处?!

  • val默认字段将所有构造函数参数转换为public readonly()
  • 生成toString(),equals()hashcode()使用所有的构造params用于在每个方法的方法
  • 生成与含有适当的同名伴侣对象apply()unapply()方法,它基本上只是一个便捷构造允许在不使用所述实例化new的关键字,并且通过生成缺省的选项包裹的提取器tuple的情况下类参数.

编辑:(案例)类的示例编译器输出(从scalatutorial.de复制)

一个简单的scala类定义

class A1(v1: Int, v2: Double)
Run Code Online (Sandbox Code Playgroud)

被编译为java代码

public class A1 extends java.lang.Object implements scala.ScalaObject {
  public A1(int, double);
}    
Run Code Online (Sandbox Code Playgroud)

类似的案例类

case class A2(v1: Int, v2: Double)
Run Code Online (Sandbox Code Playgroud)

被编译为以下java类

public class A2 extends java.lang.Object implements 
scala.ScalaObject,scala.Product,java.io.Serializable {
  public static final scala.Function1 tupled();
  public static final scala.Function1 curry();
  public static final scala.Function1 curried();
  public scala.collection.Iterator productIterator();
  public scala.collection.Iterator productElements();
  public double copy$default$2();
  public int copy$default$1();
  public int v1();
  public double v2();
  public A2 copy(int, double);
  public int hashCode();
  public java.lang.String toString();
  public boolean equals(java.lang.Object);
  public java.lang.String productPrefix();
  public int productArity();
  public java.lang.Object productElement(int);
  public boolean canEqual(java.lang.Object);
  public A2(int, double);
}


public final class A2$ extends scala.runtime.AbstractFunction2 
implements scala.ScalaObject {
  public static final A2$ MODULE$;
  public static {};
  public scala.Option unapply(A2);
  public A2 apply(int, double);
  public java.lang.Object apply(java.lang.Object, java.lang.Object);
}   
Run Code Online (Sandbox Code Playgroud)


aba*_*yuk 6

Scala案例类为您实现了大量的样板,并且将所有构造函数参数自动公开为vals就是其中之一.

如果您尝试val在常规课程中避免使用s,请执行以下操作:

trait MyTrait {
  def myVal: String
}

class MyClass(myVal: String) extends MyTrait
Run Code Online (Sandbox Code Playgroud)

编译器将显示错误消息,MyClass必须是抽象的,因为它不会覆盖myVal方法,但添加valvar类构造函数参数将解决问题.