Scala学习笔记--提取器unapply
提取器就是一个带有unapply方法的对象。你可以把unapply方法当做是伴生对象中apply方法的反向操作。
apply方法接收构造参数,然后将他们变成对象。
而unapply方法接受一个对象,然后从中取值--通常这些值就是当初用来构造该对象的值。
转自崔鹏飞的博客 博文地址:http://blog.csdn.net/cuipengfei1/article/details/33353159
实在想不到什么动词可以当做脱衣服来讲了,所以从现在开始这系列博文就叫做Desugar Scala了。除非哪天才思泉涌,又想到了新词:)
开始正文。
名字叫做unapply和unapplySeq的方法在Scala里也是有特殊含义的。
我们前面说过case class在做pattern match时很好用,而除case class之外,有unapply或unapplySeq方法的对象在pattern match时也有很好的应用场景。
比如这段代码:
1
2
3
|
|
我们定义了一个unapply方法,用来计算平方根。 我们可以像调用普通方法一样的调用它:
1
2
|
|
这样会得到36的平方根:6。实际上返回值是Some(6)。
上面的方式是对unapply的浪费,unapply真正的好处是这样的:
1
2
3
4
5
|
|
这样我们无需显式调用unapply方法,而把是它用在pattern match中,让编译器替我们调用它。
当我们写下这段pattern match的代码时,编译器其实替我们做了好几件事:
- 调用unapply,传入number
- 接收返回值并判断返回值是None,还是Some
- 如果是Some,则将其解开,并将其中的值赋值给n(就是case Square(n)中的n)
这段代码反编译出来是这个样子的:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
|
如果没有unapply方法和pattern match语法之间的这种结合,我们自己写代码要写成什么样子呢?
或许会比上面反编译的代码简单一些,但是显式地调用开平方的方法,用if else来判断Option,以及将真正的返回值从Option里面解出来这三件事是免不掉的。
unapplySeq和unapply的作用很是类似,例如这样:
1
2
3
4
5
6
|
|
我们定义一个unapplySeq方法,用逗号作为分隔符来把字符串拆开。
然后我们可以这样应用它:
1
2
3
4
5
6
7
8
|
|
另一个unapplySeq例子
object Doman{ def apply(parts : String*):String ={ parts.reverse.mkString("."); } def unapplySeq(str:String):Option[Seq[String]] = { Some(str.split("\\.").reverse); } def myMatch(str:String){ str match{ case Doman("org","acm") => println("acm.org"); case Doman("com","sun","java") => println("java.sun.com"); case Doman("net",_*) =>println("a .net doman") } } def main(args : Array[String]):Unit = { val str1 = "acm.org"; val str2 = "java.sun.com"; val str3 = "a.b.c.net"; myMatch(str1); myMatch(str2); myMatch(str3); } }
输出结果:
acm.org
java.sun.com
a .net doman
与上面的例子很是类似,不过编译器在这里替我们做的事情更多了:
- 调用unapplySeq,传入namesString
- 接收返回值并判断返回值是None,还是Some
- 如果是Some,则将其解开
- 判断解开之后得到的sequence中的元素的个数是否是三个
- 如果是三个,则把三个元素分别取出,赋值给first,second和third
如果没有unapplySeq方法和pattern match语法之间的这种结合,我们自己写代码来做这五件事会显得很是繁琐。