我有一个非常简单的问题-在Scala中创建对象时,何时应应用new关键字?当我们尝试仅实例化Java对象时,是这样吗?
Answers:
new
当您要引用class
自己的构造函数时,请使用关键字:
class Foo { }
val f = new Foo
new
如果要引用伴随对象的apply
方法,请省略:
class Foo { }
object Foo {
def apply() = new Foo
}
// Both of these are legal
val f = Foo()
val f2 = new Foo
如果您创建了案例类:
case class Foo()
Scala秘密地为您创建了一个伴随对象,将其变成了:
class Foo { }
object Foo {
def apply() = new Foo
}
所以你可以做
f = Foo()
最后,请记住,没有规则说伴侣apply
方法必须是构造函数的代理:
class Foo { }
object Foo {
def apply() = 7
}
// These do different things
> println(new Foo)
test@5c79cc94
> println(Foo())
7
而且,由于您提到了Java类:是的-Java类很少具有带有apply
方法的伴侣对象,因此必须使用new
和实际类的构造函数。
当我们尝试仅实例化Java对象时,是这样吗?
一点也不。当您new
在scala中省略时,通常有两种情况。使用单例对象(通常用于存储静态函数并作为一种工厂,类似于您在Java中可能看到的对象):
scala> object LonelyGuy { def mood = "sad" }
defined module LonelyGuy
scala> LonelyGuy
res0: LonelyGuy.type = LonelyGuy$@3449a8
scala> LonelyGuy.mood
res4: java.lang.String = sad
对于案例类(实际上,在下面是类+对象= 伴随模式,例如,具有相同名称的类和对象):
scala> case class Foo(bar: String)
defined class Foo
scala> Foo("baz")
res2: Foo = Foo(baz)
因此,当您使用简单的类时,规则与Java相同。
scala> class Foo(val bar: String)
defined class Foo
scala> new Foo("baz")
res0: Foo = Foo@2ad6a0
// will be a error
scala> Foo("baz")
<console>:8: error: not found: value Foo
Foo("baz")
另外,scala中有一个匿名类,可以这样构造:
scala> new { val bar = "baz" }
res2: java.lang.Object{val bar: java.lang.String} = $anon$1@10ee5b8
scala> res2.bar
res3: java.lang.String = baz
当我们尝试仅实例化Java对象时,是这样吗?
使用基于Dotty的 Scala 3(应于2020年中发布,八年后):永远不会。
Scala 3将在该线程中删除“ new
”
即使没有实现apply方法,Creator应用程序也允许使用简单的函数调用语法来创建类的实例。
例:
class StringBuilder(s: String) {
def this() = this(s)
}
StringBuilder("abc") // same as new StringBuilder("abc")
StringBuilder() // same as new StringBuilder()
Creator应用程序概括了迄今为止仅为案例类提供的功能,但是实现该功能的机制略有不同。
代替自动生成的apply方法,我们向函数调用添加了新的可能的解释f(args)
。