Scala第三章学习笔记

时间:2020-12-11 05:26:20

换行后的左大括号造成的问题:

class FooHolder
{
def foo()
{
println("foo was called")
}
}

Scala认为def foo()这行代码定义了一个抽象方法。这是因为它没有捕捉到后面的大括号,认定def foo()是完整的一行语句。当编译时,它认为这是一个洗呢匿名代码块,应该在类构建过程中执行。

解决办法:加一条新的编码规定,要求所有的方法定义使用"="语法。

trait FooHolder2{
def foo() : Unit =
{
println("foo2 was called")
}
}

空悬的操作符和括号表达式

"大字符串聚合"是空悬操作符能够帮到编译器的例子,指你试图创建一个很大的、无法在一行里完整定义的字符串。

java例子:

class test{
private int x = 5;
private String foo() {
return "HAI"
+ x
+ "ZOMG"
+ "\n";
}
}

在Scala下就会编译失败,可以把操作符放到行末,从而知道后面还有更多内容

class test{
private int x = 5;
private String foo() {
return "HAI" +
x +
"ZOMG" +
"\n";
}
}

也可以增加括号

class test{
private int x = 5;
private String foo() {
return ("HAI"
+ x
+ "ZOMG"
+ "\n";)
}

Scala的特质是线性化的,所以如果想要使用覆盖的方法,可以在实例化对象的时候混入父类,而不需要定义新的类。

trait Animal {def talk : String}
trait Mammal extends Animal
trait Cat {def talk = "Meow"} val x = new Mammal with Cat x.talk

执行结果如下:

Scala第三章学习笔记

注解优化 @tailrec @switch注解

使用@switch注解

Scala第三章学习笔记

编译器给出了警告语句。因为模式匹配无法优化,编译不过。

@tailre 注解用于确保可以对方法执行尾递归优化,是把最后一句语句调用自身的函数转换为不占用栈控件,而是类似传统的while或for循环那样执行。JVM本身不支持,所以依赖于Scala编译器来执行优化。

要优化尾递归调用,Scala编译器需要以下条件。

(1)方法必须是final或私有。方法不能多态。

(2)方法必须注明返回类型。

(3)方法必须在其某个分支的最后一句调用自身。

广度优先搜索算法是搜索图或树的一种算法:先检查顶层元素,然后是这些元素的最近邻居,然后是最近邻居的最近邻居,据此类推,直到找到你想要的元素。

def search(start : Node, p : Node => Boolean) = {
@tailrec
def loop(nodeQueue : List[Node],visited : Set[Node]) : Option[Node] =
nodeQueue match{
case head :: tail if p(head) =>
some(head)
case head :: tail if !visited.contains(head) =>
loop(tail ++ head.edges, visited + head)
case head :: tail ==>
loop(tail,visited)
case head Nil =>
None
}
loop(List(start),Set())
}