我刚刚在互联网上浏览了一些Scala教程,并注意到在一些示例中,对象是在示例的开头声明的。

Scala中类和对象的区别是什么?


当前回答

博士tl;

class C定义一个类,就像在Java或c++中一样。 object O创建一个单例对象O作为某个匿名类的实例;它可以用来保存与某些类的实例没有关联的静态成员。 对象O扩展了T,使对象O成为性状T的实例;你可以在任何地方传递O,期望传递T。 如果存在类C,则对象C是类C的伴生对象;注意,伴生对象并不自动是C的实例。

也可以参考Scala文档了解对象和类。

对象作为静态成员的宿主

大多数情况下,您需要一个对象来保存可用的方法和值/变量,而不必首先实例化某个类的实例。 这种用法与Java中的静态成员密切相关。

object A {
  def twice(i: Int): Int = 2*i
}

然后,您可以使用A.twice(2)调用上述方法。

如果twice是某个类a的成员,那么你需要先创建一个实例:

class A() {
  def twice(i: Int): Int = 2 * i
}

val a = new A()
a.twice(2)

您可以看到这是多么的冗余,因为twice不需要任何特定于实例的数据。

对象作为特殊的命名实例

您还可以使用对象本身作为类或trait的一些特殊实例。 当你这样做的时候,你的对象需要扩展一些特征,以成为它的一个子类的实例。

考虑下面的代码:

object A extends B with C {
  ...
}

该声明首先声明了一个匿名(不可访问)类,它扩展了B和C,并实例化了这个类的一个名为a的实例。

这意味着A可以传递给期望类型为B或C的对象的函数,或者带有C的B。

对象的附加特征

Scala中还存在一些对象的特殊特性。 我建议阅读官方文档。

def apply(…)启用A(…)常用的无方法名语法 Def unapply(…)允许创建自定义模式匹配提取器 如果伴随同名的类,则对象在解析隐式参数时承担特殊角色

其他回答

在Scala中定义一个对象就像在Java中定义一个只有静态方法的类。然而,在Scala中,对象可以扩展另一个超类,实现接口,并像类的实例一样传递。(所以它就像类的静态方法,但更好)。

类是一个定义,一个描述。它根据其他类型的方法和组合来定义类型。

一个对象是一个单例——一个类的实例,它保证是唯一的。对于代码中的每个对象,将创建一个匿名类,该类继承于您声明object要实现的任何类。这个类不能从Scala源代码中看到——尽管可以通过反射得到它。

对象和类之间有一种关系。如果一个对象具有相同的名称,则称其为类的伴生对象。当这种情况发生时,双方都可以访问另一方的私有可见性方法。但是,这些方法不会自动导入。您要么必须显式地导入它们,要么以类/对象名作为前缀。

例如:

class X {
  // class X can see private members of object X
  // Prefix to call
  def m(x: Int) = X.f(x)

  // Import and use
  import X._
  def n(x: Int) = f(x)

  private def o = 2
}

object X {
  private def f(x: Int) = x * x

  // object X can see private members of class X
  def g(x: X) = {
    import x._
    x.o * o // fully specified and imported
   }
}

形式上的区别

不能为对象提供构造函数参数 对象不是类型-不能使用new操作符创建实例。但它可以有字段、方法、扩展超类和混合特征。

用法上的区别:

Scala没有静态方法或字段。相反,你应该使用object。您可以与相关类一起使用它,也可以不使用它。在第一种情况下,它被称为同伴对象。你必须: 对类和对象使用相同的名称 把它们放在同一个源文件中。 要创建一个程序,你应该在object中使用main方法,而不是在class中。 对象Hello { def main(args: Array[String]) { println(“Hello, World !”) } } 你也可以像在java中使用单例对象一样使用它。           

Object是一个类,但它已经是一个实例,因此不能调用new ObjectName。另一方面,Class只是类型,它可以通过调用new ClassName()作为实例。

类就像其他语言中的任何其他类一样。定义类就像任何其他语言一样,只是在语法上有所不同。

class Person(val name: String)
val me = new Person("My name")

然而,object是一个只有单个对象的类。这使得它很有趣,因为它可以用于使用伴生对象创建类的静态成员。这个伴随对象可以访问类定义的私有成员,并且它与您正在定义的类具有相同的名称。

class Person(var name: String) {

  import Person._

  def hi(): String = sayHello(name)
}

object Person {
  private def sayHello(name: String): String = "Hello " + name
}

val me = new Person("My name")
me.hi()

另外,值得注意的一点是对象类是惰性创建的,这是另一个重要的点。除非在代码中需要,否则这些不会被实例化。

如果您正在为JDBC定义连接创建,您可以在对象内部创建它们以避免重复,就像我们在Java中对单例对象所做的那样。