Scala语言入门
Scala语言入门
Scala(发音为/ˈskɑːlə, ˈskeɪlə/)是一门多范式的编程语言,设计初衷是要集成面向对象编程和函数式编程的各种特性。洛桑联邦理工学院的Martin Odersky于2001年基于Funnel的工作开始设计Scala。Java平台的Scala于2003年底/2004年初发布。
特点:
- 同样运行在JVM上,可以与现存程序同时运行。
- 可直接使用Java类库。
- 同Java一样静态类型。
- 语法和Java类似,比Java更加简洁(简洁而并不是简单),表达性更强。
- 同时支持面向对象、函数式编程。
- 比Java更面向对象。
关注点:
- 类型推断、不变量、函数式编程、高级程序构造。
- 并发:actor模型。
- 和现有Java代码交互、相比Java异同和优缺。
和Java关系:
1 |
|
测试代码:Scala分支
等待替换repo,还没手敲
阅读:
环境配置
Scala需要依赖Java,访问这里查看特定Scala版本依赖的Java版本。这里选择,最新的JDK17配合Scala2.13.6。
Windows中下载安装配置环境变量:
- 类似于java配置
SCALA_HOME
为安装目录。 - 添加
%SCALA_HOME%\bin
到path环境变量。
Linux中类似,可以使用包管理器,但如果依赖版本不严格一致的话,需要官网下载对应版本安装即可。
也有交互式执行环境:
1 |
|
交互式执行环境中的传统艺能:
1 |
|
暂时不管项目配置,还是单文件编译执行为主,项目开发肯定要以包的形式组织可以使用IntelliJ IDEA开发,使用maven或者sbt进行项目配置。
使用VSCode编辑器,安装插件Scala Syntax (official)和Scala (Mentals)。
新建文件HelloScala.scala
。
1 |
|
可以使用插件CodeRunner直接快捷键运行。也可以在命令行编译为字节码后再运行:
1 |
|
或者直接运行scala源文件:
1 |
|
和java如出一辙。
如果编译的话会生成2个.class
字节码文件,HelloScala.class
和HelloScala$.class
。都是字节码但是不能通过java
直接运行。但对于HelloWorld这个例子来说,java源代码编译而成的字节码是可以通过scala
命令运行的。
原因是没有引入Scala的库,添加classpath
就可以通过java执行scala编译成的字节码了:
1 |
|
使用Java Decompiler反编译字节码到java源文件可以看到引入Scala库的逻辑。并且:
- scala源文件中的
HelloScala
对象编译后成为了一个类,但对象本身编译后就是生成的另一个类HelloScala$
类的单例对象HelloScala$.MODULE$
,称之为伴生对象。 HelloScala$
有一个main
实例方法,HelloScala
类的静态方法通过这个单例对象转调这个实例方法。完成打印。- Scala比Java更面向对象。
IDEA环境配置
使用IntelliJ IDEA:
创建Maven项目,JDK版本17。
安装插件:Scala。在IDEA里Preferences/Plugins安装scala插件,在线下载慢的话,可以去jerkins官网下载插件到本地然后from disk安装 。
Maven项目默认用Java写,在
main/
目录下新建目录scala/
,文件夹上右击Mark directory as, 然后将目录标记为Source Root。这样甚至可以在同一个项目中混用Scala和Java源文件,并互相调用。
需要能够添加scala源文件,右键项目,添加框架支持(Add Framworks Support),配置Scala SDK,选择,然后就可以右键添加Scala源文件了。
⚠️注意:到这里如果new文件还没出现scala class
🍬尝试解决方案:
在项目的目录下,找到.idea文件下,再从里面找到modules文件夹,里面有个以.iml为后缀的文件,打开它,新增下面这段代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21新增下面这段代码,name指的是你自己安装的scala版本,最后你再右键的时候可以发现能new出来一个Scala文件啦
<orderEntry type="library" name="scala-sdk-2.13.0" level="application" />
以下是整个.iml为后缀文件的参考案例:
<?xml version="1.0" encoding="UTF-8"?>
<module org.jetbrains.idea.maven.project.MavenProjectsManager.isMavenModule="true" type="JAVA_MODULE" version="4">
<component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_5">
<output url="file://$MODULE_DIR$/target/classes" />
<output-test url="file://$MODULE_DIR$/target/test-classes" />
<content url="file://$MODULE_DIR$">
<sourceFolder url="file://$MODULE_DIR$/src/main/java" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/main/resources" type="java-resource" />
<sourceFolder url="file://$MODULE_DIR$/src/main/scala" isTestSource="false" />
<sourceFolder url="file://$MODULE_DIR$/src/test/java" isTestSource="true" />
<excludeFolder url="file://$MODULE_DIR$/target" />
</content>
<orderEntry type="jdk" jdkName="1.8" jdkType="JavaSDK" />
<orderEntry type="sourceFolder" forTests="false" />
<orderEntry type="library" name="scala-sdk-2.12.10" level="application" />
</component>
</module>添加包,添加Scala类,选择对象,编辑源码。
1 |
|
- Ctrl + Shift + F10运行。
- 可以看到执行的命令是
java
在classpath
中引入了Scala的jar
包形式的库。 - 调用java的类库:
1 |
|
语法含义:
1 |
|
object
关键字创建的伴生对象,可以理解为替代Java的static
关键字的方式,将静态方法用单例对象的实例方法做了替代,做到了更纯粹的面向对象。
仅仅测试理解语法既可以单文件编写VSCode编译执行也可以用IDEA管理项目,影响不大。
再用一个等价的类定义来认识和区别一下Scala和Java:
java:
1 |
|
scala:
1 |
|
Scala库源码与API文档:
- 官网上下载下载Scala源码,解压到Scala安装目录或任意位置。
- IntelliJ IDEA进入反编译的源码后选择右上角附加源码,选择源码的
src/
目录。 - 安装包里面已经有了文档,没有的话可以单独下载。
- Scala依赖Java,某些类型就是Java的包装,库中有一部分java源码。
变量与数据类型
注释:
- 和java一样
//
单行/* */
多行/** */
文档,方法或者类前面,便于scaladoc
生成文档。
变量和常量:
1 |
|
因为Scala的函数式编程要素,所以一个指导意见就是能用常量就不要用变量。
- 声明变量时,类型可以省略,编译器会自动推导。
- 静态类型,类型经过给定或推导确定后就不能修改。
- 变量和常量声明时,必须有初始值。
- 变量可变,常量不可变。
- 引用类型常量,不能改变常量指向的对象,可以改变对象的字段。
- 不以
;
作为语句结尾,scala编译器自动识别语句结尾。
标识符命名规范:
- 字母下划线开头,后跟字母数字下划线,和C/C++/Java一样。
- 操作符开头,且只包含(+-*/#!等),也是有效的标识符。这样用会用什么奇怪的好处吗?答案是灵活到天顶星的运算符重载。
- 用反引号包括的任意字符串,即使是同39个Scala关键字同名也可以。有点奇怪的用法,尚不知道为什么。
1 |
|
关键字:
package import class obejct trait extends with type for
private protected abstract sealed final implicit lazy override
try catch finlly throw
if else match case do while for return yield
def var val
this super
new
true false null
- 其中Java没有的关键字:
object trait with implicit match yield def val var
字符串:
- 类型:
String
+
号连接*
字符串乘法,复制一个字符串多次printf
格式化输出- 字符串插值:
s"xxx${varname}"
前缀s
模板字符串,前缀f
格式化模板字符串,通过$
获取变量值,%
后跟格式化字符串。 - 原始字符串:
raw"rawstringcontents${var}"
,不会考虑后跟的格式化字符串。 - 多行字符串:
""" """
。 - 输出:
print printf println ...
1 |
|
输入:
StdIn.readLine()
StdIn.readShort() StdIn.readDouble
import scala.io.StdIn
1 |
|
读写文件:
1 |
|
数据类型:
- java基本类型
char byte short int long float double boolean
。 - java基本类型对应包装类型:
Charater Byte Short Integer Long Float Double Boolean
。 - java中不是纯粹的面向对象。
- Scala吸取了这一点,所有数据都是对象,都是
Any
的子类。 Any
有两个子类:AnyVal
值类型AnyRef
引用类型。- 数值类型都是
AnyVal
子类,和Java数值包装类型都一样,只有整数在scala中是Int
、字符是Char
有点区别。 StringOps
是java中String
类增强,AnyVal
子类。Unit
对应java中的void
,AnyVal
子类。用于方法返回值的位置,表示方法无返回值,Unit
是一个类型,只有一个单例的对象,转成字符串打印出来为()
。Void
不是数据类型,只是一个关键字。Null
是一个类型,只有一个单例对象null
就是空引用,所有引用类型AnyRef
的子类,这个类型主要用途是与其他JVM语言互操作,几乎不在Scala代码中使用。Nothing
所有类型的子类型,也称为底部类型。它常见的用途是发出终止信号,例如抛出异常、程序退出或无限循环。
整数类型:都是有符号整数,标准补码表示。
Byte
1字节Short
2字节Int
4字节Long
8字节- 整数赋初值超出表示范围报错。
- 自动类型推断,整数字面值默认类型
Int
,长整型字面值必须加L
后缀表示。 - 直接向下转换会失败,需要使用强制类型转换,
(a + 10).toByte
。
浮点类型:
Float
IEEE 754 32位浮点数Double
IEEE 754 64位浮点数- 字面值默认
Double
字符类型:
- 同java的
Character
,2字节,UTF-16编码的字符。 - 字符常量:
''
- 类型
Char
- 转义:
\t \n \r \\ \" \'
etc
布尔类型:true false
空类型:
Unit
无值,只有一个实例,用于函数返回值。Null
只有一个实例null
,空引用。Nothing
确定没有正常的返回值,可以用Nothing来指定返回值类型。好像意思是抛异常时返回Nothing,不是特别懂。
1 |
|
数据类型转换:
自动类型提升:多种数据类型混合运算,自动提升到精度最大的数据类型。
高精度赋值到低精度,直接报错。
除了图中的隐式类型转换,都需要强制类型转换。
Byte Short Char
计算时会直接提升为Int
。Boolean
不能参与整数浮点运算,不能隐式转换为整数。
强制类型转换:
toByte toInt toChar toXXXX
'a'.toInt
2.7.toInt
- 数值与String的转换:
"" + n
"100".toInt
"12.3".toFloat
12.3".toDouble.toInt
- 整数强转是二进制截取,整数高精度转低精度可能会溢出,比如
128.toByte
。
Scala标准库:
Int
Double
这些数据类型对应于Java中的原始数据类型,在底层的运行时不是一个对象,但Scala提供了从这些类型到scala.runtime.RichInt/RichDouble/...
的(低优先级)隐式类型转换(在Perdef
中定义),从而提供了非原始类型具有的对象操作。- 基本类型都是默认导入的,不需要显式导入,位于包
scala
中。还有scala.Predef
对象也是自动导入。 - 其他需要导入的包:
scala.collection
集合。scala.collection.immutable
不可变数据结构,比如数组、列表、范围、哈希表、哈希集合。scala.collection.mutable
可变数据结构,数组缓冲、字符串构建器、哈希表、哈希集合。scala.collection.concurrent
可变并发数据结构,比如字典树。
scala.concurrent
原始的并发编程。scala.io
输入输出。scala.math
基本数学操作。scala.sys
操作系统交互。scala.util.matching
正则。- 标准库中的其他部分被放在独立的分开的库中。可能需要单独安装,包括:
scala.reflect
反射API。scala.xml
xml解析、操作、序列化。scala.collection.parallel
并行集合。scala.util.parsing
parser的组合子,什么东西?scala.swing
java的GUI框架Swing的封装。- 定义了一些别名给常用的类,比如
List
是scala.collection.immutable.List
的别名,也可以理解为默认导入? - 其他别名可能是底层平台JVM提供的,比如
String
是java.lang.String
的别名。
运算符
运算符:
- 和Java基本相同。
- 算术运算:
+ - * / %
,+
可以用于一元正号,二元加号,还可以用作字符串加法,取模也可用于浮点数。没有自增和自减语法++ --
。 - 关系运算:
== != < > <= >=
- 逻辑运算:
&& || !
,&& ||
所有语言都支持短路求值,scala也不例外。 - 赋值运算:
= += -= *= /= %=
- 按位运算:
& | ^ ~
- 移位运算:
<< >> >>>
,其中<< >>
是有符号左移和右移,>>>
无符号右移。 - scala中所有运算符本质都是对象的方法调用,拥有比C++更灵活的运算符重载。
自定义运算符:
- Scala中运算符即是方法,任何具有单个参数的方法都可以用作中缀运算符,写作中缀表达式的写法。
10.+(1)
即是10 + 1
。 - 定义时将合法的运算符(只有特殊符号构成的标识符)作为函数名称即可定义。
运算符优先级:
- 当一个表达式使用多个运算符时,将根据运算符的第一个字符来评估优先级。内置的运算符和自定义运算符都是函数,遵守同样的规则。
1 |
|
- 比如下面两个表示等价:
1 |
|
上面都是粗浅的理解,以下引用自Scala2.13标准 - 06表达式 - 12前缀中缀和后缀操作一节:
词法:
1 |
|
- 可以看到前缀运算符只有
- + ! ~
含义是正负号、逻辑非、按位取反。 - 表达式是通过运算符和操作数构建的。
前缀运算:
- 仅有
+ - ! ~
,等价于后缀的方法调用.+() .-() .!() .~()
- 前缀运算符的操作数是原子的,比如
-sin(x)
被解析为-(sin(x))
。这不同于一般的函数,如果定义一个相同含义的negate
函数,那么negate sin(x)
会被解析为sin
是操作符,而negate
和(x)
是它的操作数。
后缀运算:
- 后缀的运算符可以是任意的标识符,所有的
e op
被解释为e.op
。原来还可以这样!
中缀运算:
- 一个中缀运算符可以是任意的标识符,中缀运算符
- 中缀运算符的优先级定义如下:递增顺序,优先级由运算符首字符确定,同一行拥有同等优先级。
1 |
|
- 运算符的结合性由运算符的尾字符确定,以
:
结尾的字符为右结合,其他的都是左结合。后面的集合操作运算符中有例子。 - 优先级和结合性决定了一个表达式的语义,也即是它的组合方式。具体规则:
- 多个中缀运算符同时出现,优先级更高的更显绑定到操作数。
- 如果有多个连续的同一优先级的中缀运算符(同一优先级结合性必定相同),那么按照结合性绑定,左结合从左往右,右结合从右往左。
- 后缀运算符总是比中缀运算符优先级更低。比如
e1 op1 e2 op2
总是解释为(e1 op1 e2) op2
。
- 中缀的左结合的运算符可能包含多个参数,
e op (e1, e2, ..., en)
总是被解释为e.op(e1, e2, ..., en)
。 - 对于中缀表达式
e1 op e2
,如果op
左结合那么被解释为e1.op(e2)
,如果右结合并且是它的参数是传名参数那么被解释为e2.op(e1)
,如果参数是值传递,那么解释为{val x = e2; e2.op(x)}
,其中x是一个新名称。后面的集合操作有例子。 - 前面所说的前缀运算、后缀运算、中缀运算对应的运算符分别是一元前置运算符、一元后置运算符、二元后置运算符,不存在其他类型,比如三元的条件运算符。
赋值运算符:
- 赋值运算符是指以
=
结尾的运算符。除了同样以=
开始,和>= <= !=
之外。 - 赋值运算符在当其他的解释都非法时会有特殊对待。
- 比如说
l += r
,如果没有显式定义的+=
运算符,那么会被解释为l = l + r
。 - 这种重新解释发生在满足以下两个条件的情况下:
l
没有+=
运算符。l = l + r
是类型正确的,通常情况下这意味着l
是一个左值,并且l + r
是合法的(定义了+
或者可隐式转换为定义了+
的类型)。
试验与总结:
- 内置前缀运算符有
+ - ! ~
对少量内置类型提供支持,自定义一元前缀请使用unary_
加上要定义的运算符名称,并且不要加参数列表()
。而且这东西真的可以加参数列表,但我并没有找到用运算符形式调用的方式。 - 自定义一元后置运算符(即定义为空参数列表)需要引入
scala.language.postfixOps
才能使用后缀运算符形式调用。当然用函数调用形式调用总是没有语法问题的。同样去掉参数列表后就可以后置使用。 - 二元后置运算符,是最常用的自定义运算符。优先级上面有,结合性以是否
:
结尾确定。需要特别注意的是右结合和比如C++中重载赋值运算符是有区别的,需要了解。 - Scala中内置的赋值运算符返回空
()
,所以其实是不能连续赋值的,赋值类运算符算是被特殊对待了。并且因为推崇函数式编程风格,能定义为val
则定义为val
也就不能连续赋值。因为变量都是引用变量,定义=
变成了没有道理也不可能的一件事情,所以实践中也只能一条语句只做一次赋值。 - Scala不能定义
=
运算符,但可以定义复合赋值,并且做了处理,某些情况下只要类设计得好定义了运算类运算符就不需要多去定义复合赋值了。所以说如果真要定义复合赋值返回值也应该返回()
吗?也许是的。 - 多参数的方法其实也可以写成中缀形式,
e op (e1, e2, ..., en)
。 - 对于类方法来说,运算符是函数,函数也是运算符。
例子:可窥见其灵活程度。
1 |
|
实践指南:
- 一元前缀和一元后缀运算符定义时不加参数列表,运算符形式使用。
- 在复杂表达式中使用一元前缀和后缀运算符使用时最好加括号表明优先级,不然在复杂表达式中编译器可能难以区分这是一元的还是二元的。至少定义了一元前置和后置
-
的类中无法像a - -
和- - a
这样来用。 - 二元运算符定义只给一个参数,运算符形式使用。
- 参数多于1个时不要通过运算符形式使用,但如果很清晰的话其实也无妨。
- 函数也是运算符,非特殊符号运算符形式使用也可以很有用,表达能力很强,比如
1 to 10
。 - 不要滥用,用到都搞不清谁是运算符谁是操作数就不好了。
- 经验尚浅,还需多实践。
控制流
if-else
:
1 |
|
- scala中特殊一点,
if-else
语句也有返回值,也就是说也可以作为表达式,定义为执行的最后一个语句的返回值。 - 可以强制要求返回
Unit
类型,此时忽略最后一个表达式的值,得到()
。 - 多种返回类型的话,赋值的目标变量类型需要指定为具体公共父类,也可以自动推断。
- scala中没有三元条件运算符,可以用
if (a) b else c
替代a ? b : c
。 - 嵌套条件同理。
for
循环,也叫for
推导式:
范围遍历:
for(i <- 1 to 10) {}
,其中1 to 10
是Int
一个方法调用,返回一个Range
。范围
1 to 10
1 until 10
是包含右边界和不包含右边界的范围,也可以直接用Range
类。范围步长
1 to 10 by 2
。范围也是一个集合,也可以遍历普通集合:
for(i <- collection) {}
循环守卫:即循环保护式,或者叫条件判断式,循环守卫为
true
则进入循环体内部,为
fasle
则跳过,类似于
continue
- 写法:
1
2for(i <- collection if condition) {
}- 等价于:
1
2
3
4if (i <- collection) {
if (condition) {
}
}嵌套循环同理。嵌套循环可以将条件合并到一个for中:
- 标准写法:
1
2
3
4
5for (i <- 1 to 4) {
for (j <- 1 to 5) {
println("i = " + i + ", j = " + j)
}
}- 等价写法:
1
2
3for (i <- 1 to 4; j <- 1 to 5) {
println("i = " + i + ", j = " + j)
}- 典型例子,乘法表:
1
2
3
4for (i <- 1 to 9; j <- 1 to i) {
print(s"$j * $i = ${i * j} \t")
if (j == i) println()
}循环中的引入变量,但不是循环变量:
1
2
3for (i <- 1 to 10; j = 10 - i) {
println("i = " + i + ", j = " + j)
}循环条件也可以用
{}
- 上面的引入变量循环等价于:
1
2
3
4
5
6for {
i <- 1 to 10
j = 10 - i
} {
println("i = " + i + ", j = " + j)
}循环同样有返回值,返回值都是空,也就是
Unit
实例()
。循环中同样可以用
yield
返回,外面可以接住用来操作,循环暂停,执行完后再继续循环。就像Ruby/Python。
1
val v = for (i <- 1 to 10) yield i * i // default implementation is Vector, Vector(1, 4, 9, 16, 25, 36, 49, 64, 81, 100)
while
和do while
:
- 为了兼容java,不推荐使用,结果类型是
Unit
。 - 不可避免需要声明变量在循环外部,等同于循环内部对外部变量造成了影响,所以不推荐使用。
1 |
|
循环中断:
Scala内置控制结构去掉了
break continue
关键字,为了更好适应函数式编程,推荐使用函数式风格解决。使用
breakable
结构来实现break continue
功能。循环守卫可以一定程度上替代
continue
。可以用抛出异常捕获的方式退出循环,替代break
1
2
3
4
5
6
7
8
9try {
for (i <- 0 to 10) {
if (i == 3)
throw new RuntimeException
println(i)
}
} catch {
case e: Exception => // do nothing
}可以使用Scala中的
Breaks
类中的break
方法(只是封装了异常捕获),实现异常抛出和捕获。
1
2
3
4
5
6
7
8import scala.util.control.Breaks
Breaks.breakable(
for (i <- 0 to 10) {
if (i == 3)
Breaks.break()
println(i)
}
)
函数式编程
不同范式对比:
- 面向过程:按照步骤解决问题。
- 面向对象:分解对象、行为、属性,通过对象关系以及行为调用解决问题。耦合低,复用性高,可维护性强。
- 函数式编程:面向对象和面向过程都是命令式编程,但是函数式编程不关心具体运行过程,而是关心数据之间的映射。纯粹的函数式编程语言中没有变量,所有量都是常量,计算过程就是不停的表达式求值的过程,每一段程序都有返回值。不关心底层实现,对人来说更好理解,相对地编译器处理就比较复杂。
- 函数式编程优点:编程效率高,函数式编程的不可变性,对于函数特定输入输出是特定的,与环境上下文等无关。函数式编程无副作用,利于并行处理,所以Scala特别利于应用于大数据处理,比如Spark,Kafka框架。
函数定义:
1 |
|
- 函数式编程语言中,函数是一等公民(可以像对象一样赋值、作为参数返回值),可以在任何代码块中定义函数。
- 一般将定义在类或对象中(最外层)的函数称为方法,而定义在方法中(内层)的称为函数。广义上都是函数。
- 返回值用
return
返回,不写的话会使用最后一行代码作为返回值。 - 无返回值
Unit
时可以用return
可以用return ()
可以不返回。 - 其他时候只需要返回值是返回值类型的子类对象就行。
术语说明:
- java中不提函数的说法,而是说类或者实例方法,不涉及一般化的函数。
- 函数式编程中的函数二字来源于数学上的函数,也就是映射,集合和集合之间的关系,强调数据之间的映射关系。
- 而编程语言中的函数,也包括scala中的函数定义都是指的一个完成特定功能的子程序(subroutine),并不等同于数学意义上的函数。
函数参数:
可变参数,类似于Java,使用数组包装。
def f4(str:String*): Unit = {}
。- 如果除了可变参数还有其他参数,需要将可变参数放在末尾。
- 可变参数当做数组来使用。
参数默认值:
def f5(name: String = "alice"): Unit = {}
- 和C++一样,默认参数可以不传,默认参数必须全部放在末尾。
带名称传参:
- 调用时带名称。
1
2
3
4
5
6def f6(name: String, age: Int = 20, loc: String = "BeiJing"): Unit = {
println(s"name ${name}, age ${age}, location ${loc}")
}
f6("Bob")
f6("Alice", loc = "Xi'An")
f6("Michael", 30)- 不给名称的就是按顺序赋值。
- 调用时带名参数必须位于实参列表末尾。
- 和默认参数一起使用会很方便,比如有多个默认参数,但只想覆盖其中一个。
函数至简原则:
能省则省。
最后一行代码会作为返回值,可以省略
return
。函数体只有一行代码的话,可以省略花括号。
如果返回值类型能够自动推断那么可以省略。
如果函数体中用
return
做返回,那么返回值类型必须指定。如果声明返回
Unit
,那么函数体中使用return
返回的值也不起作用。如果期望是无返回值类型,那么可以省略
=
。这时候没有返回值,函数也可以叫做过程。【2.13.0已废弃,能编过不过会提示。】无参函数如果声明时没有加
()
,调用时可以省略()
。【如果声明时有()
调用也可以省略,不过2.13.3废弃了。】不关心函数名称时,函数名称和def也可以省略,去掉返回值类型,将=修改为=>定义为匿名函数。
1
val fun = (name: String) => { println("name") }
匿名函数:
没有名称的函数,可以被赋值给一个量。也叫lambda表达式
val fun = (name: String) => { println("name") }
匿名函数定义时不能有函数的返回值类型。
简化原则:
- 参数的类型可以省略,如果可以根据高阶函数形参自动推导。
- 类型省略之后如果只有一个参数,那么可以省略参数列表的
()
,name => println(name)
。 - 匿名函数函数体只要一行,那么
{}
可以省略。 - 如果参数只出现一次,则参数可以省略,后面出现的参数用
_
代替,println(_)
也是一个lambda,表示name => {println(name)}
。 - 如果可以推断出当前传入的
println
是一个函数体,而不是函数调用语句,那么可以省略下划线。也就是省略了转调,直接将函数名称作为参数传递。
1
2
3
4
5
6
7def f(func: String => Unit): Unit = {
func("alice")
}
f((name: String) => { println(name) })
f((name) => println(name))
f(println(_))
f(println)例子:省得太极端就没有可读性了。
1 |
|
高阶函数:
- 三种形式:函数作为值传递、函数作为参数、函数作为返回值。
- 作为值传递:经过赋值之后在底层变成一个lambda对象。
1 |
|
- 函数作为参数,上面展示过了。可以传匿名函数、函数名称、lambda对象。
1 |
|
- 函数作为返回值:
1 |
|
- 现在就可以套娃了,比如定义一个返回一个返回函数的函数的函数。
高阶函数举例:
- 使用特定操作处理数组元素,得到新数组。也就是集合处理的map(映射)操作。
1 |
|
- 套娃:
1 |
|
- 上面的例子经过极致简写:只能说类型推导也太强大了。内层函数可以使用外层函数的参数。
1 |
|
- 柯里化之后:
1 |
|
函数柯里化和闭包:重点。
闭包:如果一个函数,访问到了它的外部(局部)变量的值,那么这个函数和他所处的环境,称为闭包。
- 闭包的定义:
在计算机科学中,闭包(英语:Closure),又称词法闭包(Lexical Closure)或函数闭包(function closures),是在支持头等函数的编程语言中实现词法绑定的一种技术。闭包在实现上是一个结构体,它存储了一个函数(通常是其入口地址)和一个关联的环境(相当于一个符号查找表)。环境里是若干对符号和值的对应关系,它既要包括约束变量(该函数内部绑定的符号),也要包括自由变量(在函数外部定义但在函数内被引用),有些函数也可能没有自由变量。闭包跟函数最大的不同在于,当捕捉闭包的时候,它的自由变量会在捕捉时被确定,这样即便脱离了捕捉时的上下文,它也能照常运行。捕捉时对于值的处理可以是值拷贝,也可以是名称引用,这通常由语言设计者决定,也可能由用户自行指定(如C++)。
- 因为外层调用结束返回内层函数后,经过堆栈调整(比如在C中主调或者被调清理),外层函数的参数已经被释放了,所以内层是获取不到外层的函数参数的。为了能够将环境(函数中用到的并非该函数参数的变量和他们的值)保存下来(需要考虑释放问题,可以通过GC可以通过对象生命周期控制,GC是一个常见选择),这时会将执行的环境打一个包保存到堆里面。
函数柯里化(Currying):将一个参数列表的多个参数,变成多个参数列表的过程。也就是将普通多参数函数变成高阶函数的过程。
- 定义:
在计算机科学中,柯里化(英语:Currying),又译为卡瑞化或加里化,是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数而且返回结果的新函数的技术。在直觉上,柯里化声称“如果你固定某些参数,你将得到接受余下参数的一个函数”。柯里化是一种处理函数中附有多个参数的方法,并在只允许单一参数的框架中使用这些函数。
- scala中的柯里化函数定义:
1 |
|
递归:
- 太常见了,不用过多介绍。
- 方法调用自身。
- 递归要有结束逻辑。
- 调用自身时,传递参数要有规律。
- scala中递归定义函数必须声明返回值类型,因为无法通过推导获得。
- 纯函数式语言比如Haskell,连循环都没有,很多操作都需要通过递归来做,性能比较依赖尾递归优化。
- scala中的尾递归优化例子:
1 |
|
控制抽象:
- 值调用:按值传递参数,计算值后再传递。多数语言中一般函数调用都是这个方式,C++还存在引用传递。
- 名调用:按名称传递参数,直接用实参替换函数中使用形参的地方。能想到的只有C语言中的带参宏函数,其实并不是函数调用,预处理时直接替换。
- 例子:
1 |
|
- 应用:使用传名参数实现一个函数相当于while的功能。
1 |
|
惰性加载:
- 当函数返回值被声明为
lazy
时,函数的执行将会被推迟,知道我们首次对此取值,该函数才会被执行。这种函数成为惰性函数。
1 |
|
- 有点像传名参数,但懒加载只是推迟求值到第一次使用时,而不是单纯替换。
包管理
关于Scala面向对象:
- Scala的面向对象思想源自Java,很多概念是一致的。
- 语法和java不同,补充了更多功能。
包:
package name
作用:
- 区分相同名字类,避免名称冲突。
- 类很多时,分模块管理。
- 访问权限控制。
命名:包名称只能是常规的标识符(字母数字下划线,数字不能开头)。同样
.
作为不同层级分割符,整体作为包名。命名规范:一般情况下按照如下规则命名
com.company.projectname.modulename
,视项目规定而定,只是一个名称而已。scala中的两种包管理方式:
- 第一种,java风格,每个源文件声明一个包,写在源文件最上方。但源文件位置不需要和包名目录层级一致,只代表逻辑层级关系,不像java一样源文件也必须按照包名目录层级关系放置。当然惯例是和java一样按照包名目录层级来放置。
- 第二种,用
{}
嵌套风格定义包:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19package com {
// code in com package
object Outer {
var name = "Outer"
}
package inner {
// code in com.inner package
package scala {
// code in com.innner.scala package
object Inner {
def main(args: Array[String]):Unit = {
println(Outer.name)
Outer.name = "Inner"
println(Outer.name)
}
}
}
}
}- 嵌套风格好处:
- 一个源文件可以声明多个并列的最顶层的包。
- 子包中的类可以访问父包中的内容,无需导入。但外层是不能直接访问内层的,需要导入。
- 如果单文件VsCode测试嵌套包,而不是用IDE的话,那定义了包就不能直接执行了,需要
scalac
先编译,并指定入口类运行。编译后的字节码文件和java一样会自动按照包层级关系排列。
1
2scalac PackageManagement.scala
scala com.inner.scala.Inner
包对象:
- 为scala包定义一个同名的单例包对象,定义在包对象中的成员,作为其对应包下的所有类和对象的共享变量,可以被直接访问,无需导入。
- 关键字
package object
,需要和包在同一层级下。比如为com.inner
包定义包对象的话,必须在com
包中,定义形式package obejct inner { ... }
。
包的导入:
1 |
|
- 可以在任意位置导入(作用于代码块),可以设置别名,可以选择性导入想要导入的内容,可以屏蔽某个类。
- 所有scala源文件默认导入:
1 |
|
面向对象
类定义:
- 回顾java中,如果是
public
向外公开的,那么必须和文件名一致,也只能有一个。不写访问修饰符则可以定义多个,包访问权限。 - scala中没有
public
关键字,默认就是公有,不能加public
,一个文件可以写多个类,不要求和文件名一致。
1 |
|
- 访问修饰符可以是:
private
protected
private [pacakgeName]
,默认就是公有,不需要加。 - 成员如果需要Java Bean规范的getter和setter的话可以加
@scala.beans.BeanProperty
相当于自动创建,不需要显式写出。 - 成员给初值
_
会赋默认值,scala中定义变量必须赋值,可以这样做。值类型的值0,引用则是null
。定义常量的话不能用_
,因为只能初始化一次,编译器会提示。
封装:
- Java的封装:私有化,提供getter和setter。
- scala中考虑到Java太冗余了,脱裤子放屁一样。scala中的公有属性,底层实际为
private
,并通过get方法obj.field()
和set方法obj.field_=(value)
对其进行操作。所以scala不推荐设置为private
。如果需要和其他框架互操作,必须提供Java Bean规范的getter和setter的话可以加@scala.beans.BeanProperty
注解。
访问权限:
- Java中
private protected public
和默认包访问权限。 - scala中属性和方法默认公有,并且不提供
public
关键字。 private
私有,类内部和伴生对象内可用。protected
保护权限,scala中比java中严格,只有同类、子类可访问,同包无法访问。【因为java中说实话有点奇怪】private [pacakgeName]
增加包访问权限,在包内可以访问。
构造器:
- 包括主构造器和辅助构造器。
1 |
|
- 例子:
1 |
|
特点:
- 主构造器写在类定义上,一定是构造时最先被调用的构造器,方法体就是类定义,可以在类中方法定义的同级编写逻辑,都是主构造器一部分,按顺序执行。
- 辅助构造器用
this
定义。 - 辅助构造器必须直接或者间接调用主构造器,调用其他构造必须位于第一行。
- 主构造器和辅助构造器是重载的方法,所以参数列表不能一致。
- 可以定义和类名同名方法,就是一个普通方法。
主构造器中形参三种形式:不使用任何修饰,
1
var
修饰,
1
val
修饰。
- 不使用任何修饰那就是一个形参,但此时在类内都可以访问到这个变量。逻辑上不是一个成员(报错信息这么写),但是可以访问,WTF???
- 使用
var val
修饰那就是定义为类成员,分别是变量和常量,不需要也不能在类内再定义一个同名字段。调用时传入参数就直接给到该成员,不需要再显式赋值。 - 主构造器中的
var val
成员也可以添加访问修饰符。 - 不加参数列表相当于为空,
()
可以省略。 - 主构造器的访问修饰符添加到参数列表
()
前。
实践指南:
- 推荐使用scala风格的主构造器
var val
修饰参数的编写方法,而不要被Java毒害! - 如果需要多种重载的构造器那么就添加新的的辅助构造器。
- 推荐使用scala风格的主构造器
1 |
|
继承:
class ChildClassName[(argList1)] extends BaseClassName[(args)] { body }
- 子类继承父类属性和方法。
- 可以调用父类构造器,但感觉好像很局限,子类中只可能调用到主构造或者辅助构造中的其中一个构造器。那如果父类有多种构造方式,子类想继承也没有办法?只能是其中一种。
- 不考虑太多负担,按照scala惯用写法来写起来还是挺轻松的。
多态:
- java中属性静态绑定,根据变量的引用类型确定,方法是动态绑定。
- 但scala中属性和方法都是动态绑定。就属性而言,其实也不应该在子类和父类中定义同名字段。
- 同java一样,所有实例方法都是虚方法,都可以被子类覆写。
override
关键字覆写。- scala中属性(字段)也可以被重写,加
override
关键字。
抽象类:
abstract calss ClassName
- 抽象属性:
val/var name: Type
,不给定初始值。 - 抽象方法:
def methodName(): RetType
,只声明不实现。 - 子类如果没有覆写全部父类未定义的属性和方法,那么就必须定义为抽象类。老生常谈了。
- 重写非抽象方法属性必须加
override
,重写抽象方法则可以不加override
。 - 子类调用父类中方法使用
super
关键字。 - 子类重写父类抽象属性,父类抽象属性可以用
var
修饰,val var
都可以。因为父类没有实现嘛,需要到子类中来实现。 - 如果是重写非抽象属性,则父类非抽象属性只支持
val
,不支持var
。因为var
修饰为可变量,子类继承后可以直接使用修改,没有必要重写。val
不可变才有必要重写。 - 实践建议是重写就加
override
,都是很自然的东西,理解就好,不必纠结于每一个细节。
匿名子类;
- 和java如出一辙。重写所有抽象字段和方法。
1 |
|
伴生对象(Companion Object):
- 取代
static
语义。 - 编译后其实会生成两个类,伴生对象是伴生类(类名为对应类后加
$
符号)的单例对象。 obejct
,名称和类一致,必须放同一个文件,前面已经说过了。- 常见用法:构造器私有化,用伴生对象中的工厂方法。和静态工厂方法使用起来也没有什么区别。
- 伴生对象实现
apply
方法后调用时可以省略.apply
,直接使用className(args)
。库中很多这种用法创建实例,是一个语法糖。 - 测试伴生对象时就在该对象内定义
main
函数编译时会出现的奇怪的访问权限问题。可能对包含入口的伴生对象做了特殊处理,具体细节尚不清楚。最好将main
定义在单独的伴生对象内。
Trait(特征/特质):
- 替代java接口的概念。但比接口更为灵活,一种实现多继承的手段。
- 多个类具有相同的特征时,就可以将这个特征提取出来,用继承的方式来复用。
- 用关键字
trait
声明。
1 |
|
- 引入/混入(mixin)特征:
- 有父类
class extends baseClass with trait1 with trait2 ... {}
- 没有父类
class extends trait1 with trait2 ... {}
- 有父类
- 其中可以定义抽象和非抽象的属性和方法。
- 匿名子类也可以引入特征。
- 特征和基类或者多个特征中重名的属性或方法需要在子类中覆写以解决冲突,最后因为动态绑定,所有使用的地方都是子类的字段或方法。属性的话需要类型一致,不然提示不兼容。方法的话参数列表不一致会视为重载而不是冲突。
- 如果基类和特征中的属性或方法一个是抽象的,一个非抽象,且兼容,那么可以不覆写。很直观,就是不能冲突不能二义就行。
- 多个特征和基类定义了同名方法的,就需要在子类重写解决冲突。其中可以调用父类和特征的方法,此时
super.methodName
指代按照顺序最后一个拥有该方法定义的特征或基类。也可以用super[baseClassOrTraitName].methodName
直接指代某个基类的方法,注意需要是直接基类,间接基类则不行。 - 也就是说基类和特征基本是同等地位。
- 例子:
1 |
|
- 特征的继承:
trait childTrait extends baseTrait
- 特征的菱形继承解决方式:转换为线性的继承链条,在前面的成为基类,后面的成为子类。
- 例子:
1 |
|
- 其实特征的多继承和C++的多继承已经很像了,只是名称冲突的解决方式不一样,菱形继承的解决方式也不一样,而且不能访问间接基类。
- scala单继承多实现,实现体现在特征上。基类主要用于一个对象比较核心比较本质的部分上。
- 继承特征与类的区别:特征构造时不能给参数。其他都是同样的,都可以实现多态。
自身类型(self type):
- 可实现依赖注入的功能。
- 一个类或者特征指定了自身类型的话,它的对象和子类对象就会拥有这个自身类型中的所有属性和方法。
- 是将一个类或者特征插入到另一个类或者特征中,属性和方法都就像直接复制插入过来一样,能直接使用。但不是继承,不能用多态。
- 语法,在类或特征中:
_: SelfType =>
,其中_
的位置是别名定义,也可以是其他,_
指代this
。插入后就可以用this.xxx
来访问自身类型中的属性和方法了。 - 注入进来的目的是让你能够使用,可见,提前使用应该拥有的属性和方法。最终只要自身类型和注入目标类型同时被继承就能够得到定义了。
- 例子:
1 |
|
运行时类型识别RTTI:
- 判断类型:
obj.isInstanceOf[T]
,确切匹配的类型或者父类都返回true。 - 转换类型:
obj.asInstance[T]
,转换为目标类型。 - 获取类名:
classOf[T]
,得到类对应的Class
对象Class[T]
,转字符串结果是class package.xxx.className
。 - 获取对象的类:
obj.getClass
枚举类:
- 继承
Enumeration
。 - 用
Value
类型定义枚举值。
1 |
|
应用类:
- 继承
App
,包装了main
方法,就不需要显式定义main
方法了,可以直接执行。
1 |
|
定义类型别名:type SelfDefineType = TargetType
。
密封类: sealed
,子类只能定义在同一个文件内。
集合
Java集合:
- 三大类型:列表
List
、集合Set
、映射Map
,有多种不同实现。
Scala集合三大类型:
序列
Seq
,集合Set
,映射Map
,所有集合都扩展自Iterable
。对于几乎所有集合类,都同时提供
可变和不可变
版本。
- 不可变集合:
scala.collection.immutable
- 可变集合:
scala.collection.mutable
- 两个包中可能有同名的类型,需要注意区分是用的可变还是不可变版本,避免冲突和混淆。
- 不可变集合:
对于不可变集合,指该集合长度数量不可修改,每次修改(比如增删元素)都会返回一个新的对象,而不会修改源对象。
可变集合可以对源对象任意修改,一般也提供不可变集合相同的返回新对象的方法,但也可以用其他方法修改源对象。
建议:操作集合时,不可变用操作符,可变用方法。操作符也不一定就会返回新对象,但大多是这样的,还是要具体看。
scala中集合类的定义比java要清晰不少。
不可变集合:
scala.collection.immutable
包中不可变集合关系一览:- 不可变集合没有太多好说的,集合和映射的哈希表和二叉树实现是肯定都有的,序列中分为随机访问序列(数组实现)和线性序列(链表实现),基本数据结构都有了。
Range
是范围,常用来遍历,有语法糖支持1 to 10 by 2
10 until 1 by -1
其实就是隐式转换加上方法调用。- scala中的
String
就是java.lang.String
,和集合无直接关系,所以是虚箭头,是通过Perdef
中的低优先级隐式转换来做到的。经过隐式转换为一个包装类型后就可以当做集合了。 Array
和String
类似,在图中漏掉了。- 此类包装为了兼容java在scala中非常常见,scala中很多类型就是对java类型的包装或者仅仅是别名。
- scala中可能会推荐更多地使用不可变集合。能用不可变就用不可变。
可变集合:
scala.collection.mutable
包中可变集合关系一览:
- 序列中多了
Buffer
,整体结构差不多。
不可变和可变:
- 不可变指的是对象大小不可变,但是可以修改元素的值(不能修改那创建了也没有用对吧),需要注意这一点。而如果用了
val
不变量存储,那么指向对象的地址也不可变。 - 不可变集合在原集合上个插入删除数据是做不到的,只能返回新的集合。
泛型:
- 集合类型大多都是支持泛型,使用泛型的语法是
[Type]
,不同于java的<Type>
。
不可变数组:
- 访问元素使用
()
运算符,通过apply/update
方法实现,源码中的实现只是抛出错误作为存根方法(stab method),具体逻辑由编译器填充。
1 |
|
- 可以看到自定义运算符可以非常灵活,规定如果运算符首尾有
:
那么:
一定要指向对象。 - 下标越界当然会抛出异常,使用前应该检查。
- 通过
Predef
中的隐式转换为一个混入了集合相关特征的包装类型从而得以使用scala的集合相关特征,Array
类型中并没有相关混入。
可变数组:
- 类型
ArrayBuffer
。
1 |
|
- 推荐:不可变集合用运算符,可变集合直接调用对应方法。运算符容易迷惑。
- 更多方法查看文档和源码用到去找就行。
- 可变数组和不可变数组可以调用方法互相转换。
二维数组:
- 就是数组的数组。
- 使用
Array.ofDim[Type](firstDim, secondDim, ...)
方法。
1 |
|
不可变列表:
List
,抽象类,不能直接new
,使用伴生对象apply
传入元素创建。List
本身也有apply
能随机访问(做了优化),但是不能update
更改。foreach
方法遍历。- 支持
+: :+
首尾添加元素。 Nil
空列表,::
添加元素到表头。- 常用
Nil.::(elem)
创建列表,换一种写法就是10 :: 20 :: 30 :: Nil
得到结果List(10, 20, 30)
,糖是真滴多! - 合并两个列表:
list1 ::: list2
或者list1 ++ list2
。
可变列表:
- 可变列表
ListBuffer
,和ArrayBuffer
很像。 final
的,可以直接new
,也可以伴生对象apply
传入元素创建(总体来说scala中更推荐这种方式)。- 方法:
append prepend insert remove
- 添加元素到头或尾:
+=: +=
- 合并:
++
得到新的列表,++=
合并到源上。 - 删除元素也可以用
-=
运算符。 - 具体操作很多,使用时阅读文档即可。
不可变集合:
- 数据无序,不可重复。
- 可变和不可变都叫
Set
,需要做区分。默认Set
定义为immutable.Set
别名。 - 创建时重复数据会被去除,可用来去重。
- 添加元素:
set + elem
- 合并:
set1 ++ set2
- 移除元素:
set - elem
- 不改变源集合。
可变集合:
- 操作基于源集合做更改。
- 为了与不可变集合区分,
import scala.collection.mutable
并用mutable.Set
。 - 不可变集合有的都有。
- 添加元素到源上:
set += elem
add
- 删除元素:
set -= elem
remove
- 合并:
set1 ++= set2
- 都很简单很好理解,多看文档和源码就行。
不可变映射:
Map
默认就是immutable.Map
别名。- 两个泛型类型。
- 基本元素是一个二元组。
1 |
|
可变映射:
mutable.Map
- 不可变的都支持。
1 |
|
元组:
(elem1, elem2, ...)
类型可以不同。- 最多只能22个元素,从
Tuple1
定义到了Tuple22
。 - 使用
_1 _2 _3 ...
访问。 - 也可以使用
productElement(index)
访问,下标从0开始。 ->
创建二元组。- 遍历:
for(elem <- tuple.productIterator)
- 可以嵌套,元组的元素也可以是元组。
集合通用属性和方法:
- 线性序列才有长度
length
、所有集合类型都有大小size
。 - 遍历
for (elem <- collection)
、迭代器for (elem <- collection.iterator)
。 - 生成字符串
toString
mkString
,像Array
这种是隐式转换为scala集合的,toString
是继承自java.lang.Object
的,需要自行处理。 - 是否包含元素
contains
。
衍生集合的方式:
- 获取集合的头元素
head
(元素)和剩下的尾tail
(集合)。 - 集合最后一个元素
last
(元素)和除去最后一个元素的初始数据init
(集合)。 - 反转
reverse
。 - 取前后n个元素
take(n) takeRight(n)
- 去掉前后n个元素
drop(n) dropRight(n)
- 交集
intersect
- 并集
union
,线性序列的话已废弃用concat
连接。 - 差集
diff
,得到属于自己、不属于传入参数的部分。 - 拉链
zip
,得到两个集合对应位置元素组合起来构成二元组的集合,大小不匹配会丢掉其中一个集合不匹配的多余部分。 - 滑窗
sliding(n, step = 1)
,框住特定个数元素,方便移动和操作。得到迭代器,可以用来遍历,每个迭代的元素都是一个n个元素集合。步长大于1的话最后一个窗口元素数量可能个数会少一些。
集合的简单计算操作:
- 求和
sum
求乘积product
最小值min
最大值max
maxBy(func)
支持传入一个函数获取元素并返回比较依据的值,比如元组默认就只会判断第一个元素,要根据第二个元素判断就返回第二个元素就行xxx.maxBy(_._2)
。- 排序
sorted
,默认从小到大排序。从大到小排序sorted(Ordering[Int].reverse)
。 - 按元素排序
sortBy(func)
,指定要用来做排序的字段。也可以再传一个隐式参数逆序sortBy(func)(Ordering[Int].reverse)
- 自定义比较器
sortWith(cmp)
,比如按元素升序排列sortWith((a, b) => a < b)
或者sortWith(_ < _)
,按元组元素第二个元素升序sortWith(_._2 > _._2)
。 - 例子:
1 |
|
- 简单操作还是太少了,不足以应对复杂的需求。
集合高级计算函数:
- 大数据的处理核心就是映射(map)和规约(reduce)。
- 映射操作(广义上的map):
- 过滤:自定义过滤条件,
filter(Elem => Boolean)
- 转化/映射(狭义上的map):自定义映射函数,
map(Elem => NewElem)
- 扁平化(flatten):将集合中集合元素拆开,去掉里层集合,放到外层中来。
flatten
- 扁平化+映射:先映射,再扁平化,
flatMap(Elem => NewElem)
- 分组(group):指定分组规则,
groupBy(Elem => Key)
得到一个Map,key根据传入的函数运用于集合元素得到,value是对应元素的序列。
- 过滤:自定义过滤条件,
- 规约操作(广义的reduce):
- 简化/规约(狭义的reduce):对所有数据做一个处理,规约得到一个结果(比如连加连乘操作)。
reduce((CurRes, NextElem) => NextRes)
,传入函数有两个参数,第一个参数是第一个元素(第一次运算)和上一轮结果(后面的计算),第二个是当前元素,得到本轮结果,最后一轮的结果就是最终结果。reduce
调用reduceLeft
从左往右,也可以reduceRight
从右往左(实际上是递归调用,和一般意义上的从右往左有区别,看下面例子)。 - 折叠(fold):
fold(InitialVal)((CurRes, Elem) => NextRes)
相对于reduce
来说其实就是fold
自己给初值,从第一个开始计算,reduce
用第一个做初值,从第二个元素开始算。fold
调用foldLeft
,从右往左则用foldRight
(翻转之后再foldLeft
)。具体逻辑还得还源码。从右往左都有点绕和难以理解,如果要使用需要特别注意。
- 简化/规约(狭义的reduce):对所有数据做一个处理,规约得到一个结果(比如连加连乘操作)。
- 以上:
1 |
|
集合应用案例:
- Map的默认合并操作是用后面的同key元素覆盖前面的,如果要定制为累加他们的值可以用
fold
。
1 |
|
- 经典案例:单词计数:分词,计数,取排名前三结果。
1 |
|
- 单词计数案例扩展,每个字符串都可能出现多次并且已经统计好出现次数,解决方式,先按次数合并之后再按照上述例子处理。
1 |
|
- 当然这并不高效,更好的方式是利用上已经统计的频率信息。
1 |
|
队列:
- 可变队列
mutable.Queue
- 入队
enqueue(Elem*)
出队Elem = dequeue()
- 不可变队列
immutable.Queue
,使用伴生对象创建,出队入队返回新队列。
并行集合(Parllel Collection):
- 使用并行集合执行时会调用多个线程加速执行。
- 使用集合类前加一个
.par
方法。 - 具体细节待补。
- 依赖
scala.collection.parallel.immutable/mutable
,2.13版本后不再在标准库中提供,需要单独下载,暂未找到编好的jar的下载地址,从源码构造需要sbt,TODO。
模式匹配
match-case
中的模式匹配:
- 用于替代传统C/C++/Java的
switch-case
结构,但补充了更多功能,拥有更强的能力。 - 语法:(Java中现在也支持
=>
的写法了)
1 |
|
- 每一个case条件成立才返回,否则继续往下走。
case
匹配中可以添加模式守卫,用条件判断来代替精确匹配。
1 |
|
- 模式匹配支持类型:所有类型字面量,包括字符串、字符、数字、布尔值、甚至数组列表等。
- 你甚至可以传入
Any
类型变量,匹配不同类型常量。 - 需要注意默认情况处理,
case _
也需要返回值,如果没有但是又没有匹配到,就抛出运行时错误。默认情况case _
不强制要求通配符(只是在不需要变量的值建议这么做),也可以用case abc
一个变量来接住,可以什么都不做,可以使用它的值。 - 通过指定匹配变量的类型(用特定类型变量接住),可以匹配类型而不匹配值,也可以混用。
- 需要注意类型匹配时由于泛型擦除,可能并不能严格匹配泛型的类型参数,编译器也会报警告。但
Array
是基本数据类型,对应于java的原生数组类型,能够匹配泛型类型参数。
1 |
|
- 对于数组可以定义多种匹配形式,可以定义模糊的元素类型匹配、元素数量匹配或者精确的某个数组元素值匹配,非常强大。
1 |
|
- List匹配和Array差不多,也很灵活。还可用用集合类灵活的运算符来匹配。
- 比如使用
::
运算符匹配first :: second :: rest
,将一个列表拆成三份,第一个第二个元素和剩余元素构成的列表。
- 比如使用
- 注意模式匹配不仅可以通过返回值当做表达式来用,也可以仅执行语句类似于传统
switch-case
语句不关心返回值,也可以既执行语句同时也返回。 - 元组匹配:
- 可以匹配n元组、匹配元素类型、匹配元素值。如果只关心某个元素,其他就可以用通配符或变量。
- 元组大小固定,所以不能用
_*
。
变量声明匹配:
- 变量声明也可以是一个模式匹配的过程。
- 元组常用于批量赋值。
val (x, y) = (10, "hello")
val List(first, second, _*) = List(1, 3, 4, 5)
val List(first :: second :: rest) = List(1, 2, 3, 4)
for
推导式中也可以进行模式匹配:
- 元组中取元素时,必须用
_1 _2 ...
,可以用元组赋值将元素赋给变量,更清晰一些。 for ((first, second) <- tupleList)
for ((first, _) <- tupleList)
- 指定特定元素的值,可以实现类似于循环守卫的功能,相当于加一层筛选。比如
for ((10, second) <- tupleList)
- 其他匹配也同样可以用,可以关注数量、值、类型等,相当于做了筛选。
- 元组列表匹配、赋值匹配、
for
循环中匹配非常灵活,灵活运用可以提高代码可读性。
匹配对象:
- 对象内容匹配。
- 直接
match-case
中匹配对应引用变量的话语法是有问题的。编译报错信息提示:不是样例类也没有一个合法的unapply/unapplySeq
成员实现。 - 要匹配对象,需要实现伴生对象
unapply
方法,用来对对象属性进行拆解以做匹配。
样例类:
- 第二种实现对象匹配的方式是样例类。
case class className
定义样例类,会直接将打包apply
和拆包unapply
的方法直接定义好。- 样例类定义中主构造参数列表中的
val
甚至都可以省略,如果是var
的话则不能省略,最好加上的感觉,奇奇怪怪的各种边角简化。
对象匹配和样例类例子:
1 |
|
偏函数:
- 偏函数是函数的一种,通过偏函数我们可以方便地对参数做更精确的检查,例如偏函数输入类型是
List[Int]
,需要第一个元素是0的集合,也可以通过模式匹配实现的。 - 定义:
1 |
|
- 通过一个变量定义方式定义,
PartialFunction
的泛型类型中,前者是参数类型,后者是返回值类型。函数体中用一个case
语句来进行模式匹配。上面例子返回输入的List
集合中的第二个元素。 - 一般一个偏函数只能处理输入的一部分场景,实际中往往需要定义多个偏函数用以组合使用。
- 例子:
1 |
|
异常处理
scala异常处理整体上的语法和底层处理细节和java非常类似。
Java异常处理:
- 用
try
语句包围要捕获异常的块,多个不同catch
块用于捕获不同的异常,finally
块中是捕获异常与否都会执行的逻辑。
1 |
|
scala异常处理:
try
包围要捕获异常的内容,catch
仅仅是关键字,将捕获异常的所有逻辑包围在catch
块中。finally
块和java一样都会执行,一般用于对象的清理工作。- scala中没有编译期异常,所有异常都是运行时处理。
- scala中也是用
throw
关键字抛出异常,所有异常都是Throwable
的子类,throw
表达式是有类型的,就是Nothing
。Nothing
主要用在一个函数总是不能正常工作,总是抛出异常的时候用作返回值类型。 - java中用了
throws
关键字声明此方法可能引发的异常信息,在scala中对应地使用@throws[ExceptionList]
注解来声明,用法差不多。
1 |
|
隐式转换
前面说了很多了,编译器做隐式转换的时机:
- 编译器第一次编译失败时,会在当前环境中查找能让代码编译通过的方法,将类型隐式转换,尝试二次编译。
隐式函数:
- 函数定义前加上
implicit
声明为隐式转换函数。 - 当编译错误时,编译器会尝试在当前作用域范围查找能调用对应功能的转换规则,这个过程由编译器完成,称之为隐式转换或者自动转换。
1 |
|
- 在当前作用域定义时需要在使用前定义才能找到。
1 |
|
隐式参数:
- 普通方法或者函数中的参数可以通过
implicit
关键字声明为隐式参数,调用方法时,如果传入了,那么以传入参数为准。如果没有传入,编译器会在当前作用域寻找复合条件的隐式值。例子:集合排序方法的排序规则就是隐式参数。 - 隐式值:
- 同一个作用域,相同类型隐式值只能有一个。
- 编译器按照隐式参数的类型去寻找对应隐式值,与隐式值名称无关。
- 隐式参数优先于默认参数。(也就是说隐式参数和默认参数可以同时存在,加上默认参数之后其实就相当于两个不同优先级的默认参数)
- 隐式参数有一个很淦的点:
- 如果参数列表中只有一个隐式参数,无论这个隐式参数是否提供默认参数,那么如果要用这个隐式参数就应该将调用隐式参数的参数列表连同括号一起省略掉。如果调用时又想加括号可以在函数定义的隐式参数列表前加一个空参数列表
()
,那么()
将使用隐式参数,()()
将使用默认参数(如果有,没有肯定编不过),()(arg)
使用传入参数。 - 也就是说一个隐式参数时通过是否加括号可以区分隐式参数、默认参数、传入参数三种情况。
- 那么如果多参数情况下:隐式参数、默认参数、普通参数排列组合在一个参数列表中混用会怎么样呢?没有试验过,不要这么用,思考这些东西搞什么哦!
- 具体要不要加这个柯里化的空参数列表,那看习惯就行。不加可能更好一点,加了可能有点让人费解。
- 如果参数列表中只有一个隐式参数,无论这个隐式参数是否提供默认参数,那么如果要用这个隐式参数就应该将调用隐式参数的参数列表连同括号一起省略掉。如果调用时又想加括号可以在函数定义的隐式参数列表前加一个空参数列表
- 可以进一步简写隐式参数,在参数列表中直接去掉,在函数中直接使用
implicity[Type]
(Predef
中定义的)。但这时就不能传参数了,有什么用啊?相当于一个在自己作用域范围内起作用的全局量?
1 |
|
隐式类:
- scala2.10之后提供了隐式类,使用
implicit
声明为隐式类。将类的构造方法声明为隐式转换函数。 - 也就是说如果编译通不过,就可能将数据直接传给构造转换为对应的类。
- 隐式函数的一个扩展。
- 说明:
- 所带构造参数有且只能有一个。
- 隐式类必须被定义在类或者伴生对象或者包对象中,隐式类不能是顶层的。
- 同一个作用域定义隐式转换函数和隐式类会冲突,定义一个就行。
隐式解析机制的作用域:
- 首先在当前代码作用域下查找隐式实体(隐式方法、隐式类、隐式对象)。
- 如果第一条规查找隐式对象失败,会继续在隐式参数的类型的作用域中查找。
- 类型的作用域是指该类型相关联的全部伴生对象以及该类型所在包的包对象。
作用:
- 隐式函数和隐式类可以用于扩充类的功能,常用语比如内建类
Int Double String
这种。 - 隐式参数相当于就是一种更高优先级的默认参数。用于多个函数需要同一个默认参数时,就不用每个函数定义时都写一次默认值了。为了简洁无所不用其极啊真是。
泛型
泛型:
[TypeList]
,定义和使用都是。- 常用于集合类型中用于支持不同元素类型。
- 和java一样通过类型擦除/擦拭法来实现。
- 定义时可以用
+-
表示协变和逆变,不加则是不变。
1 |
|
协变和逆变:
- 比如Son和Father是父子关系,Son是子类。
- 协变(Covariance):
MyList[Son]
是MyList[Father]
的子类,协同变化。 - 逆变(Contravariance):
MyList[Son]
是MyList[Father]
的父类,逆向变化。 - 不变(Invariant):
MyList[Father] MyList[Son]
没有父子关系。
- 协变(Covariance):
- 还需要深入了解。
泛型上下限:
- 泛型上限:
class MyList[T <: Type]
,可以传入Type
自身或者子类。 - 泛型下限:
class MyList[T >: Type]
,可以传入Type
自身或者父类。 - 对传入的泛型进行限定。
上下文限定:
def f[A : B](a: A) = println(a)
等同于def f[A](a: A)(implicit arg: B[A])
- 是将泛型和隐式转换结合的产物,使用上下文限定(前者)后,方法内无法使用隐式参数名调用隐式参数,需要通过
implicitly[Ordering[A]]
获取隐式变量。 - 了解即可,可能基本不会用到。
Style Guide
官方的Style Guide中的一些建议:
- 缩进鼓励为2个,当然我上面都是用的4个。Scala中很多时候嵌套层次会很深,也鼓励这样做,模式匹配、匿名函数、循环、条件等各种嵌套,层次深了之后4空格可能会比较折磨。
- 一个表达式一行放不下要换行时,语义上不会产生歧义就行,比如一个运算符放在末尾将其必需的操作数换到下一行。
- 多参数函数调用需要换行书写时,将第一个参数放到第二行并缩进2个空格书写,而不是第一个参数放到第一行,然后缩进到对齐(典型java风格)。
1 |
|
- 仅介绍第一页的内容,也没有空去看完,以后真写得多了再去看。
sbt
上面已经简单介绍了IDEA使用Maven项目编写Scala的配置,但学习scala,官方的构建工具sbt还是必须要了解一下的。
关于SBT:
- SBT是Scala的构建工具,全称Simple Build Tool,类似于 Maven 或 Gradle。
- GETTING STARTED WITH SCALA AND SBT ON THE COMMAND LINE
- sbt Reference Manual 要使用sbt,阅读完第一章Getting Started with sbt是必要的。下面的内容皆是第一章翻译。
特性:
- 简单项目零配置。
- 用Scala源码管理项目构建。
- 精确的重编译,节省时间。
- 使用Coursier的库管理器。
- 支持Scala和Java的混合项目。
- 等等等,具体就不列了,总之一个大型项目构建系统该有的东西。
安装:
- sbt依赖Java,确保已经安装了JDK1.8或以上版本。
- 下载压缩包或者安装包,这里的版本是1.5.5。
- 解压或者安装。
- 配置环境变量
SBT_HOME
,并添加%SBT_HOME%\bin
到path环境变量,安装包的话会自动配置。
通过案例入门sbt
创建一个项目hello作为例子:
- windows上没有的命令按照含义操作即可。
1 |
|
开始sbt shell:
1 |
|
- 第一次初始化时间会很长。
- 退出shell:
1 |
|
- 编译:
1 |
|
- sbt shell中Tab可以补全。
对修改重新编译:
- 在
compile
命令(或其他命令同理)前加一个~
前缀,会进入等待状态,当项目发生修改是会自动重新编译。当然退出这个状态后就不会了。
1 |
|
创建源文件:
- 执行
~compile
并保持,创建目录src/main/scala/example
新建源文件保存就能看到编译过程了。
1 |
|
sbt shell常用操作:
help
帮助。help run
具体条目的帮助。run
运行程序。- 上下箭头切换已执行命令。
scalaVersion
scala版本。
配置修改:
- 切换当前项目的scala版本:
set ThisBuild / scalaVersion := "2.13.6"
。 session save
保存配置到build.sbt
,此时其中就会多出ThisBuild / scalaVersion := "2.13.6"
。- 编辑
build.sbt
:
1 |
|
- 重新加载配置
reload
。
测试:
- 添加ScalaTest到依赖
1 |
|
- 执行测试:
test
。 - 后续继续运行追加的测试:
~testQuick
编写测试:src/test/scala/HelloSpec.scala
1 |
|
- 测试结果当然是失败
1 |
|
- 改一下源码再测试就能通过了:
1 |
|
添加库依赖:
1 |
|
使用REPL(Read-Eval-Print Loop):
1 |
|
- 在scala的REPL环境中粘贴:
:paste
。 - 退出:
:q
修改build.sbt
创建一个子项目:
1 |
|
reload
时会自动创建目录core/
。- 列出所有子项目:
projects
- 编译子项目:
1 |
|
- 子项目添加依赖:
1 |
|
广播命令、添加依赖:
- 设置
.aggregate(...)
,这样发送到hello
的命令都会被广播到helloCore
- 使用
.dependsOn(...)
可以设置依赖,下面的设置使hello
依赖于helloCore
- 将Gigahorse的依赖移到
helloCore
。
1 |
|
使用Play JSON解析JSON:
- 添加依赖。
1 |
|
- 重载,添加文件:
core/src/main/scala/example/core/Weather.scala
1 |
|
- 修改
src/main/scala/example/Hello.scala
:
1 |
|
- 运行:
run
1 |
|
添加sbt-native-packager插件:
- 创建
project/plugins.sbt
1 |
|
- 修改
build.sbt
对Hello
项目添加.enablePlugins(JavaAppPackaging)
- 重载,本地没有执行成功,所以下面的
dist
命令也就不能用。
重载并创建.zip分发包:dist
1 |
|
应用容器化:
Docker/publishLocal
- 运行容器化后的应用:
docker run hello:0.1.0-SNAPSHOT
设置应用版本:
1 |
|
临时切换Scala版本:
++2.12.14
在Bash中直接运行sbt的命令:
1 |
|
- 这样程序运行起来会慢一些。
- 连续的开发的话,推荐使用sbt shell或者连续测试比如
~testQuick
。
new
命令:
1 |
|
- 会创建一个简单的项目,要求输入项目名时输入
hello
,会在hello/
下创建一个新项目。
sbt使用
项目的目录结构:
- base directory是包含项目的目录,这里称为项目根目录。
- sbt使用和Maven一样的源码结构,源文件路径都是基于项目根目录的相对路径。
1 |
|
- 其他
src/
中的目录会被忽略,所有隐藏目录都会被忽略。 - 源码可以被放在根目录的
hello/app.scala
,对小项目是可行的。然而一般来说,人们倾向于将项目放在src/main/
下面来保证事情能够有条理地进行。如果你自行管理定制了项目的构建的话,自定义源码的位置也是可行的。 - sbt的构建定义文件:
build.sbt
- 除此之外,
project
目录中的.scala
文件可以定义项目帮助文件和一次性的插件。
1 |
|
- 生成文件:
.class
,生成的jar
,其他文件和文档等会被默认输出到target
目录。 - 一般生成文件应该要排除在版本控制之外,在
.gitignore
中添加:
1 |
|
运行:
- 运行sbt shell:
sbt
无参数运行,进入sbt的提示符,有tab补全和执行历史。 - 编译:
compile
- 运行:
run
- 不进入sbt shell直接运行sbt命令:用
""
包起来表示是一个命令,相对来说会慢一些。
1 |
|
- 会一次执行
clean compile testOnly
,TestA TestB
是传给testOnly
的参数。 - 保存文件时自动重编译运行测试:
~testQuick
。 - 命令加上
~
后会进入循环模式,保存文件都会自动运行。回车退出。 - 常用命令:
Command | Description |
---|---|
clean | Deletes all generated files (in the target directory). |
compile | Compiles the main sources (in src/main/scala and src/main/java directories). |
test | Compiles and runs all tests. |
console | Starts the Scala interpreter with a classpath including the compiled sources and all dependencies. To return to sbt, type :quit, Ctrl+D (Unix), or Ctrl+Z (Windows). |
run argument* | Runs the main class for the project in the same virtual machine as sbt. |
package | Creates a jar file containing the files in src/main/resources and the classes compiled from src/main/scala and src/main/java. |
help command | Displays detailed help for the specified command. If no command is provided, displays brief descriptions of all commands. |
reload | Reloads the build definition (build.sbt, project/.scala, project/.sbt files). Needed if you change the build definition. |
build.sbt
build.sbt
构建定义:
- 指定sbt版本,这样就使用不同版本的sbt构建同一个项目了。如果指定的版本不可用,那么会自动下载。
1 |
|
- 构建定义(build definition):
- 包含了一系列项目(Scala中的Project类型),项目这个名词有一定的模糊性,所以其中的一个个项目一般将之称为子项目。
- 在
build.sbt
中可以定义一个在当前目录中的子项目:
1 |
|
- 项目的名称在
.setting
方法中用一个键值对定义,key是name
,值是一个字符串表示项目名称。 build.sbt
定义所有的子项目,包含一些的键值对称为setting
表达式,使用一门build.sbt DSL(本质上其实就是Scala)来定义项目。
1 |
|
看一看这门DSL的定义:
每一个都叫做一个setting expression,其中的一些又被叫做task expression。
一个setting expression包含三部分:
- 左边是key。
- 操作符,这个例子中是
:=
- 右边是setting body。
一个key的类型是
sbt.SettingKey[T] sbt.TaskKey[T] sbt.InputKey[T]
其中一者的实例,T是期望的值类型。比如
name
就绑定到了SettingKey[String]
类型,给个其他类型比如整数的话就会编译错误。在
build.sbt
中可以穿插val
lazy val
def
,但是不能有顶层object class
定义。
expression的key:
- 三种类型:
SettingKey[T]
值仅在加载子项目时计算一次,然后保持。TaskKey[T]
值被称为一个任务(task),每次都会重新计算(何时?),存在潜在的副作用。InputKey[T]
值是有命令行输入作为参数的任务,细节见Input Tasks。
- 内建的keys就是
sbt.Keys
单例伴生对象的域。build.sbt
隐式导入import sbt.Keys._
,sbt.Keys.name
就是name
,所以其实就是对这些字段做赋值。 - 自定义key:使用各自的方法
settingKey taskKey inputKey
,每个方法需要一个value的类型和描述。key的名称就是被赋值到的引用变量名称。 - 定义一个自定义key,名称是
hello
,类型是TaskKey
,对应值类型是Unit
:
1 |
|
- 所有这种定义都在设置前被求值,无论定义在文件什么位置。
- 一般来说,使用
lazy val
而不是val
来避免初始化顺序导致的问题。 - Task和Setting区别:
- Task是任务,比如
compile
package
都是sbt.Keys
中的域,同时也是sbt shell中可执行的命令。应该返回Unit
或者返回和这个任务相关的值,比如package
是TaskKey[File]
值是其创建的jar文件。 - 每一次开始一个任务,比如sbt shell中执行
compile
,sbt都会重新跑(仅)一次这个任务相关的所有任务。 - 而Setting仅仅只是一个朴素的设置项。
- Task是任务,比如
定义任务和设置:
- 使用
:=
可以将一个设置或者一项计算任务赋值。设置只会在记载项目时计算一次,任务则会在每次执行这个任务被执行时重新执行。 - 新建一个任务:
1 |
|
- 任务也在
.settings
中被赋值。 - 每次在sbt shell中执行
hello
都会执行其中println
语句。 - 定义设置的话已经说过。
- 从类型系统的视角来看,对任务赋值得到一个
Setting[Task[T]]
,对设置赋值得到Setting[T]
。T Task[T]
的区别有一层隐含的含义:一个设置不能依赖于一个任务。因为设置仅记载是求值一次不会每次都重新运行。
sbt shell中的key:
- sbt shell中可以输入任何任务名称都会运行该任务,因为这个任务名称是key。运行该任务但并不会显示运行结果值(也就是返回值,类型就是
taskKey[T]
中的T
),要显示结果值,应该使用show <task name>
而不是单纯的<task name>
。 - 如果输入设置的key的话,会显示设置的值。
- 要知道一个key的更多信息,可以使用
inspect <keyname>
。某些信息现在看起来可能不知道含义,但最顶上有类型和简要描述。
在build.sbt
中导入信息:
- 比如:
1 |
|
- 中间不能有空行。
- 如果有自动插件(
sbt.AutoPlugin
,可以从其派生实现自己的插件),那么在其中的autoImport
单例对象下的名称会被自动导入。
Bare .sbt build definition:
- 也就是裸的构建定义,设置可以被直接写到
.sbt
而不是项目的.setting(...)
调用下,称之为bare style。
1 |
|
- 这种语法推荐用来写在
ThisBuild
作用域下的设置和添加插件。后续会有作用域和插件的说明。
添加库依赖:
- 为了能够依赖第三方库,有两种方式:
- 一是将jar文件直接放在
lib/
(未管理的依赖)目录下,第二种是添加管理的依赖,通过在.setting(..)
调用中对libraryDependencies
key做+=
操作做到。
1 |
|
- 包含包的组织、包名、和包的版本。可以定义变量来复用。
%
运算符被用来从字符串构建一个模块ID。
多项目构建
多个项目定义:
- 在一次构建中编译多个相关联的子项目是很有用的,特别是其间存在依赖,想更改他们所有的时候。
- 每个子项目都有自己的目录,构建时都会生成自己的jar文件,
- 项目使用
lazy val
定义一个sbt.Project来实现。
1 |
|
- 这个
val
的不变量名称被用做子项目的ID(也即是项目名称),在sbt shell中也用来指代一个子项目。 - 后面的
in file()
调用指定他们的base directory是可选的,目录名就是他们的名称。
1 |
|
公共设置;
- 为了分离出跨子项目的设置,可以将其定义在
ThisBuild
范围下。ThisBuild
表现的像一个普通的子项目名称一样使用,其下用来定义默认值。 - 如果定义了多个子项目,并且子项目下没有定义比如
scalaVersion
这个key,就会查找ThisBuild / scalaVersion
。 - 这样定义的限制是右边的值只能是纯粹的值或者在
Global
/ThisBuild
下的设置。 - 子项目范围下没有默认值。
1 |
|
- 重载后,现在
versoin
和其他的设置在所有子项目中都会生效。
另一种定义公共设置的方式:
- 将默认设置放在
commonSettings
下,然后添加到所有子项目中.setting()
调用中。
1 |
|
项目间依赖:
- 项目间可以完全独立,也通常可能会有某种方式的依赖。
- 有两种方式的依赖:aggregate and classpath,用
.aggregate .dependsOn
定义。 - 聚合用来广播命令,类路径依赖则是指项目之间存在依赖关系。
聚合(Aggregation):
1 |
|
- 这种方式在聚合项目上运行任务时会同样在它聚合的所有子项目上运行。
- 比如上面的子项目定义,编译
root
是会同时编译util core
。 - 通过在做聚合的项目中定义设置可以修改这种默认行为:
1 |
|
- 上面设置表示在root项目上执行
update
是就不会在被聚合的子项目上执行。update / aggregate
是update作用域下的key。
类路径依赖:
- 一个项目可能依赖另一个项目的代码,通过
.dependsOn(proj1, proj2, ...)
方法调用来定义。依赖之后就会被添加到classpth,从而能够导入。
1 |
|
- 现在在
core
中就可以调用util
中的代码了。这同样也确定了代码编译顺序必然是先util
后core
。 core dependsOn(util)
意味着core
的编译配置依赖util
,也可以显式通过dependsOn(util % "compile->compile")
这种方式来指定,compile->compile
中的->
意味着依赖。因此如果是"test->compile"
就以为着core
的test
配置依赖util
的compile
配置。可以忽略后面的->config
部分意味着就是->comile
。- 一个比较有用的定义是
test->test
意味着测试core
是先测试util
。 - 可以用分好分隔:
dependsOn(util % "test->test;compile->compile")
。
项目间依赖:
在大项目中,会有许多文件,持续监视文件修改并重新编译将会消耗大量磁盘和IO资源。
sbt使用
1
trackInternalDependencies
和
1
exportToInternal
设置用来控制执行
1
compile
任务时是否触发独立子项目的编译。两个key都接受三个输入:
TrackLevel.NoTracking
TrackLevel.TrackIfMissing
TrackLevel.TrackAlways
默认是这一项。- 含义显而易见。
trackInternalDependencies
设置如果是TrackLevel.TrackIfMissing
,那么sbt将不会尝试自动编译项目间依赖,除非输出目录中的*.class
文件缺失了(或者exportJars
时jar文件缺失了)。当被设置为
TrackLevel.NoTracking
,项目将依赖将被跳过。但是classpth仍然会被添加,依赖图也会显示他们还是依赖的。这么做的目的是为了减小检查文件修改的IO负担。设置方法:
1 |
|
exportToInternal
设置允许依赖于当前项目的子项目跳过内部追踪。用在一个子项目上,用在当前不关心的依赖于其他项目的子项目,当其他项目发生修改重新编译,它也不会重新编译。
1 |
|
trackInternalDependencies
和exportToInternal
,比如当前修改项目时A,A依赖B,C依赖A,那么前者是针对B也就是当前项目依赖的那些子项目,后者针对C也就是依赖当前项目的子项目。这是我的理解,应该是这个样子!- 如果没理解错的话,项目的依赖关系(指classpth,会调用的那种依赖)应该是一棵树,而不会有环图。
默认的根项目:
- 如果没有为最顶层目录
.
定义项目,那么sbt会创建一个默认的然后在构建时聚合所有子项目。
与项目的交互:
projects
列出所有子项目。proejct
列出当前项目,project <projectname>
切换项目。运行一个任务比如compile
时是针对当前项目。- 也可以通过指定项目名称来在某个项目上运行任务:
subProjectID / <taskname>
。
公共代码:
.sbt
文件之间的定义是不共享的。- 为了能够在不同
.sbt
之间共享代码,需要在根目录的project/
下面(子目录中的是没有用的,会被忽略,只有根目录中的才会有效)定义一个或多个scala文件。后续会详述。
子项目中的.sbt
文件:
- 所有的
.sbt
文件都会被合并到一个整体的构建(build)中来,但是只在他们自己的范围内起作用。定义不会被共享。 - 比如顶层
hello
目录中初始化sbt
,.
定义根项目hello
,hello/foo/ hello/bar
分别定义项目foo和bar并且有自己的build.sbt
并把项目定义在了其中,其中定义了自己的不同版本。 - 那么执行
show version
的结果就是这样的:
1 |
|
- 所有的
build.sbt
都是整个构建的一部分,但都有自己的作用范围。 - 可以分开定义,也可以合起来定义
build.sbt
,项目很多时都定义在根目录中可能就太复杂了,定义在子目录找起来好像又挺麻烦。 - 风格选择:
- 子项目的设置在子项目的
.sbt
中定义,根build.sbt
中只定义最小的项目声明,形如:lazy val foo = (project in file("foo"))
不修改任何设置。 - 推荐是将所有项目定义全都放在根目录中的
build.sbt
,保持项目定义在一个文件中。 - 都可以,这完全取决于你。
- 子项目的设置在子项目的
任务图
任务图:
- 除了将设置视作一个个键值对,更好的比喻其实是以有向无环图(DAG)。边的方向表示在之前发生,称之为任务图(Task Graph) 。
- Setting/Task expression就是前面的在
.setting(...)
中定义了设置或者任务的表达式。
任务间依赖:
- 使用一个特殊的
.value
方法调用来解释任务之间的依赖。 - 直到非常熟悉
hello.sbt
之前,都推荐将.value
调用放在task定义块中最上方。 - 除了使用一个不变量赋值的方式,也可以使用内联的
.value
调用,更加间接,也不用去想变量名。 .value
调用会在进入task body之前被求值,这是需要非常注意的。- 测试:
1 |
|
- 此时任务
hi
就会依赖于任务update
和clean
,并且这两个任务是在进入hi
任务体前执行的,且不确定两者先后顺序,可先可后可并行。 - 在任务体中调用
.value
仅用来表明任务之间的依赖关系。 - 先编译项目,执行
hi
后会发现target/scala-2.13/clsses/
被清除,就是因为执行了clean
任务。 - 查看任务间依赖:在
Dependencies:
后可以看到。
1 |
|
- 执行
inspect tree compile
会看到compile
命令的依赖树,什么含义暂时不知。 - sbt中的构建任务依赖是自动的而不是显式定义的,如果通过
.value
定义了,那么会造成任务间依赖。
任务依赖设置:
- 在任务体定义中调用设置的
.value
就行。 - 但设置是不能依赖任务的,因为设置只在重载时执行一次,而任务一直都可以执行。
设置依赖设置:
- 可以将设置看做仅在记加载是执行一次的任务,所以设置也可以依赖设置。
- 同样其中调用
.value
就可以依赖,并且执行时求值: - 一个实际的例子:当Scala版本是2.11时讲将
Compile / scalaSource
定义到一个不同的目录。
1 |
|
再看build.sbt
DSL:
- 构建了一个设置和任务的有向无环图。
- 设置表达式编码了设置、任务以及他们之间的依赖。
- 这种结构在Make/Ant/Rake等构建工具中很常见。
- 基于流的编程,减少了重复过程,好处:
- 一个任务仅仅只会执行一次,即使它被多个任务所依赖比如
Compile / compile
。 - 基于任务图,任务引擎会安排不相关的任务并行执行。
- 关注点分离和灵活性,任务图让用户可以将任务以不同的方式连接到一起,sbt和各种插件可以提供各种各样的特性比如库依赖管理等。
总结:
- 核心就是任务图,任务之间的关系是一个有向无环图。
hello.sbt
是一个设计来面向依赖编程(dependency-oriented programming)的DSL,或者叫基于流(flow)的编程。就像Makefile。语法就像Scala,但并不等同于Scala,语义上是有区别的。- 基于流的编程的核心是:减少重复过程、并行处理和定制化。
更多内容
更多内容就不翻译了,官网上第一章也有中文翻译。需要时再去阅读,现阶段感觉都用不到,真写项目了再看不迟。
并发编程
TODO
总结
总结:
- 看起来是一门静态类型语言,提供了很其强大的类型推导,可以一定程度上实现隐式静态类型,但写起来如果高度依赖类型推导的话会和动态类型一样简洁,仅需提供少量必须的类型,只是有点牺牲可读性。
- 函数式编程很有趣。
- 语法糖太太太多了,虽然看起来更简洁了,但是读起来不一定更简单,学起来心智负担也更大。
- 运算符非常灵活,目前遇到过的运算符最灵活的语言。
- 并发编程还没有学,TODO。
- Scala语法确实有点太强大了,当然软件工程的东西都是tradeoff,写起来爽用起来复杂学起来难。
Scala是我目前学过的最舒服的语言,很多特点简直太棒了,如果此生只能选一门语言的话,那我可能真会选这门刚学了几天的语言。吸引我的点:
- 函数式编程,和集合的映射推导结合起来很有用。
- 类型推导,像动态语言用起来的感觉,但也有编译期类型检查,再加上隐式类型转换,真我全都要。
- 各种能简则简的语法糖,初看可能很诧异,习惯之后只能说去**的java,简洁而不简单。
- 运算符重载,容易被滥用,但用得好会使代码进一步简化,当然各式各样的运算符会进一步增加读代码的难度。
- 更加纯粹的面向对象,万物皆是对象。
觉得不错的话,支持一根棒棒糖吧 ୧(๑•̀⌄•́๑)૭
wechat pay
alipay