[TOC]
关于scala中的特殊方法apply,先给一个比较通俗的解释。
当对象(伴生对象)以函数的方式进行调用时,scala 会隐式地将调用改为在该对象上调用apply方法。
例如: Demo(“hello”) 实际调用的是 Demo.apply(“hello”), 因此apply方法又被称为注入方法。
apply方法常用于创建类实例的工厂方法。
case class SomeBean(name: String, age: Int)
对于这个 case 类会生成一个 SomeBean 的伴生对象 SomeBean,这个伴生对象中定义了 apply 与 unapply 方法。
下面这段代码是反编译后得到的
public Option<Tuple2<String, Object>> unapply(SomeBean x$0) {
return x$0 == null ? None..MODULE$ : new Some(new Tuple2(x$0.name(), BoxesRunTime.boxToInteger(x$0.age())));
}
public SomeBean apply(String name, int age) {
return new SomeBean(name, age);
}
apply
将对象以函数的方式进行调用时,scala会隐式地将调用改为在该对象上调用apply方法。例如XXX(“hello”)实际调用的是XXX.apply(“hello”), 因此apply方法又被称为注入方法。apply方法常用于创建类实例的工厂方法。
定义在 object 中
object Greeting{
def apply(name: String) = "Hello " + name
}
Greeting.apply(“Lucy”) //与下面的调用等价
Greeting(“Lucy”) //结果为 Hello Lucy
定义在 class 或 trait 中
首先我们来看下 scala 集合类 List 对 apply 的使用。scala 中 List 在读取数据时与 java 的 List 不同。
java 中使用 list.get(index) 取得第 index 个索引的数据,但在 scala 中只需要使用 list(index )就可以实现。
这个功能就是借助于 apply 方法实现的。例如:
List(1, 2, 3)
List("a", "b", "c")
或者是Map也有类似的用法:
Map(1 -> "a", 2 -> "b", 3 -> "c")
以上的两段代码并不是在调用 List 和 Map 的 constructor,而是在调用 List 和 Map 的伴生对象(companion objects)的 apply 方法。
下面我们来实际使用 apply 方法来显示获取特定位置的字符的需求(类似与模拟 Map 和 List 中的取值方法)
class IndexedString(val str: String) {
def apply(charIndex: Int) = str.charAt(charIndex)
}
我们可以通过 IndexedString 的 apply 方法可以通过索引读取在字符中对应的字符
val indexed = new IndexedString("Hello world")
indexed(0) //结果为H, 等价于indexed.apply(0)
scala 的函数对象
scala 中的函数对象(头等函数或first-class function)是使用 apply 的典型用法:scala 中的函数对象是scala.FunctionN 特质的实例,使用时函数对象直接调用 FunctionN 的 apply 方法。
scala> val increment: (Int) => Int = _ + 1
increment: Int => Int = <function1>
// 调用 scala.FunctionN的apply方法
scala> increment(0)
res0: Int = 1
unapply
与 apply 相对的是 unapply 方法,它的用法与 apply 类似,但其作用是用来抽取部分参数,它也称为抽取方法,主要用于模式匹配时抽取某些参数 case XXX(str) => println(str)
Scala比Java更面向对象的一个方面是Scala没有静态成员。替代品是,Scala有单例对象:singleton object。
当单例对象与某个类共享同一个名称时,他被称作是这个类的伴生对象:companion object。你必须在同一个源文件里定义类和它的伴生对象。类被称为是这个单例对象的伴生类:companion class。类和它的伴生对象可以互相访问其私有成员。
类和单例对象间的一个差别是,单例对象不带参数,而类可以。因为你不能用new关键字实例化一个单例对象,你没机会传递给它参数。每个单例对象都被作为由一个静态变量指向的虚构类:synthetic class的一个实例来实现,因此它们与Java静态类有着相同的初始化语法。Scala程序特别要指出的是,单例对象会在第一次被访问的时候初始化。
Scala 的apply 有2 张形式,一种是 伴生对象的apply ,一种是 伴生类中的apply,下面展示这2中的apply的使用。
class ApplyOperation {
}
class ApplyTest{
def apply(b:String) = println(b+":I am into spark so much!!!")
def haveATry: Unit ={
println("have a try on apply")
}
}
object ApplyTest{
def apply(i:Int) = {
println(i.toString+":I am into Scala so much")
new ApplyTest
}
}
object ApplyOperation{
def main (args: Array[String]) {
val array= Array(1,2,3,4)
val a = ApplyTest(111) //这里就是使用object 的使用
val b= ApplyTest.apply(222)
a.haveATry
a("AAA") // 这里就是 class 中 apply使用
b.apply("BBB")
new ApplyTest
new ApplyTest()
(new ApplyTest).apply("CCC")
// ApplyTest("DDD")--error object的apply方法参数是Int类型
//new ApplyTest(555)--error class的apply方法不是构造方法
ApplyTest(333)
ApplyTest.apply(444)
}
}
Scala<apply的几种用法>
1.只是一个快捷方式
首先定义个object:
object Greet {
def apply(name: String): Unit = {
println("Call From %s".format(name))
}
}
object Main {
def main(args: Array[String]): Unit = {
Greet.apply("Gpwner")
println("==========================")
Greet("pwner")
}
}
所以当调用一个Object的时候,其实就是相当于调用了这个Object的apply方法
但是apply并不等同与构造函数

2.scala会自动为case class 生成apply方法
下面的三个操作都是一样的结果:
val p0 = new Person("Frank", 23, "Blue") // normal constructor
val p1 = Person("Frank", 23, "Blue") // this uses apply
val p2 = Person.apply("Frank", 23, "Blue") // using apply manually
3、apply应用于builder模式
一个object中的apply方法不一定要返回其自身对象,比如
case class Company(name: String)
class Person(val name: String) {}
object Person {
def apply(name: String): Company = new Company(name)
}
利用这个特点我们可以将一个类写成Builder
abstract class DatabaseDriver {
// some database stuff
}
object DatabaseDriver {
def apply(config: Configuration) = config.dbType match {
case "MYSQL" => new MySqlDriver()
case "PSQL" => new PostgresDriver()
case _ => new GenericDriver()
}
}
// now I get the right version!
val mydatabase = DatabaseDriver(dbConfig)
5、在class中的apply
可以在object中有apply方法,当然也在class中定义apply方法
class Amazing {
def apply(x: String) = "Amazing %s!".format(x)
}
// look how cool this is
val amazing = new Amazing()
amazing("world")
// => Amazing world!
amazing.apply("world")
// => Amazing world!
网友评论