Scala 是一门多范式的编程语言,类似于 Java 。设计初衷是实现可伸缩的语言、并集成面向对象编程和函数式编程的各种特性
1.交互式 Scala 解释器的使用方法
为了使用交互式 Scala 解释器,你可以在打开的终端中输入命令:
su -l hadoop #密码为hadoop
scala
当出现 scala> 开始的命令行提示符时,就说明你已经成功进入解释器了。如下图所示。
1.1使用交互式 Scala 解释器
开始使用 Scala 的最简单的方式是使用交互式 Scala 解释器,只要输入 Scala 表达式, Scala 解释器会立即解释执行该语句并输出结果。当然你也可以使用如 Scala IDE 或 IntelliJ IDEA 集成开发环境。不过本教程开始还是以这种交互式 Scala 解释器为主。
使用 Scala 解释器,首先你需要下载安装 Scala 运行环境。 然后在命令行输入scala,则进入 scala 解释器
2.变量、函数的定义
2.1定义变量
Scala 定义了两种类型的变量 val 和 var ,val 类似于Java中的 final 变量,一旦初始化之后,不可以重新赋值(我们可以称它为 常变量 )。而 var 类似于一般的非 final 变量。可以任意重新赋值。
比如定义一个字符串常变量:
scala> val msg="Hello,World"
msg: String = Hello,World
这个表达式定义了一个 msg 变量,为字符串常量。它的类型为 string ( java.lang.string )。 可以看到我们在定义这个变量时并不需要像 Java 一样定义其类型, Scala 可以根据赋值的内容推算出变量的类型。这在 Scala 语言中称为“ type inference ”(类型推断)。当然如果你愿意,你也可以采用和 Java 一样的方法,明确指定变量的类型,如:
scala> val msg2:String ="Hello again,world"
msg2: String = Hello again,world
不过这样写就显得不像 Scala 风格了。此外 Scala 语句也不需要以分号结尾。 如果在命令行中需要分多行输入, Scala 解释器在新行前面显示 |,表示该行接着上一行。比如:
scala> val msg3=
| "Hello world 3rd time"
msg3: String = Hello world 3rd time
2.2.定义一些函数
Scala 既是面向对象的编程语言,也是面向函数的编程语言,因此函数在 Scala 语言中的地位和类是同等的。下面的代码定义了一个简单的函数求两个值的最大值:
scala> def max(x:Int,y:Int) : Int ={
| if (x >y) x
| else
| y
| }
max: (x: Int, y: Int)Int
Scala 函数以 def 定义,之后是函数的名称(如 max ),然后是以逗号分隔的参数。Scala 中变量类型是放在参数和变量的后面,以 : 隔开。这样做的一个好处是便于“ type inference ”。刚开始有些不习惯(如果你是 Pascal 程序员可能会觉得很亲切)。同样如果函数需要返回值,它的类型也是定义在参数的后面(实际上每个Scala函数都有返回值,只是有些返回值类型为 Unit ,类似于 void 类型)。
此外每个 Scala 表达式都有返回结果(这一点和 Java,C# 等语言不同),比如 Scala 的 if else 语句也是有返回值的,因此函数返回结果无需使用 return 语句。实际上在Scala代码中应当尽量避免使用 return 语句。函数的最后一个表达式的值就可以作为函数的结果进行返回。
同样由于 Scala 的“ type inference ”特点,本例其实无需指定返回值的类型。对于大多数函数 Scala 都可以推测出函数返回值的类型,但目前来说回溯函数(函数调用自身)还是需要指明返回结果类型的。
下面再定义个“没有”返回结果的函数(其它语言可能称这种无返回值的函数为程式)。
scala> def greet() = println("hello,world")
greet: ()Unit
greet 函数的返回值类型为 Unit ,表示该函数不返回任何有意义的值,Unit 类似于 Java 中的 void 类型。这种类型的函数主要用来获得函数的“副作用”,比如本函数的副作用是打印招呼语。
3.循环、迭代的实现
3.1.使用 while 实现
val args = Array("I","like","scala")
var i=0
while (i < args.length) {
println (args(i))
i+=1
}
这里要注意的是 Scala 不支持 ++i 和 i++ 运算符,因此需要使用 i += 1 来加一。 这段代码看起来和 Java 代码差不多,实际上 while 也是一个函数,你自然可以利用 Scala 语言的扩展性,实现 while 语句,使它看起来和 Scala 语言自带的关键字一样调用。
Scala 访问数组的语法是使用 () 而非 [] 。
这里介绍了使用 while 来实现循环,但这种实现循环的方法并不是最好的 Scala 风格,在下一步介绍使用一种更好的方法来避免通过索引来枚举数组元素。
3.2使用for和foreach来实现迭代
在实验3.4中使用 while 来实现循环,和使用 Java 实现无太大差异,而 Scala 是面向函数的语言,更好的方法是采用“函数式”风格来编写代码。比如上面的循环,使用 foreach 方法如下:
args.foreach(arg => println(arg))
该表达式,调用 args 的 foreach 方法,传入一个参数,这个参数类型也是一个函数( lambda 表达式,和 C# 中概念类似)。这段代码可以再写得精简些,你可以利用 Scala 支持的缩写形式,如果一个函数只有一个参数并且只包含一个表达式,那么你无需明确指明参数。因此上面的代码可以写成:
args.foreach( println)
Scala 中也提供了一个称为“ for comprehension ” 的功能,它比 Java 中的 for 功能更强大。“for comprehension”(可称之为 for 表达式)将在后面介绍,这里先使用 for 来实现前面的例子:
for (arg <-args)
println(arg)
4.数组的参数化
在 Scala 中, 你可以使用 new 来实例化一个类。当你创建一个对象的实例时,你可以使用数值或类型参数。如果使用类型参数,它的作用类似 Java 或 .Net 的 Generic 类型。所不同的是, Scala 使用方括号来指明数据类型参数,而非尖括号。比如:
val greetStrings = new Array[String](3)
greetStrings(0)= "Hello"
greetStrings(1)= ","
greetStrings(2)= "world!\n"
for(i <- 0 to 2)
print(greetStrings(i))
可以看到 Scala 使用 [] 来为数组指明类型化参数,本例使用 String 类型,数组使用 () 而非 [] 来指明数组的索引。其中的 for 表达式中使用到 0 to 2 ,这个表达式演示了 Scala 的一个基本规则,如果一个方法只有一个参数,你可以不用括号和 . 来调用这个方法。
因此这里的 0 to 2, 其实为 (0).to(2) 调用的为整数类型的 to 方法,to 方法使用一个参数。Scala 中所有的基本数据类型也是对象(和 Java 不同),因此 0 可以有方法(实际上调用的是 RichInt 的方法),这种只有一个参数的方法可以使用操作符的写法(不用 . 和括号),实际上 Scala 中表达式 1+2 ,最终解释为 (1).+(2)+,这也是 Int 的一个方法,和 Java 不同的是,Scala 对方法的名称没有太多的限制,你可以使用符号作为方法的名称。
这里也说明为什么 Scala 中使用 () 来访问数组元素,在 Scala 中,数组和其它普遍的类定义一样,没有什么特别之处,当你在某个值后面使用 () 时,Scala 将其翻译成对应对象的 apply 方法。因此本例中 greetStrings(1) 其实是调用 greetString.apply(1) 方法。这种表达方法不仅仅只限于数组,对于任何对象,如果在其后面使用 () ,都将调用该对象的 apply 方法。同样的如果对某个使用 () 的对象赋值,比如:
5.List、元组、Set 和 Map 的使用
5.1List
Scala 也是一个面向函数的编程语言,面向函数的编程语言的一个特点是,调用某个方法不应该有任何副作用,参数一定,调用该方法后,返回一定的结果,而不会去修改程序的其它状态(副作用)。这样做的一个好处是方法和方法之间关联性较小,从而方法变得更可靠和重用性高。使用这个原则也就意味着需要把变量设成不可修改的,这也就避免了多线程访问的互锁问题。
前面介绍的数组,它的元素是可以被修改的。如果需要使用不可以修改的序列,Scala 中提供了 Lists 类。和 Java 的 List 不同,Scala 的 Lists 对象是不可修改的。它被设计用来满足函数编程风格的代码。它有点像 Java 的 String,String 也是不可以修改的,如果需要可以修改的 String 对像,可以使用 StringBuilder 类。
比如下面的代码:
val oneTwo = List(1,2)
val threeFour = List(3,4)
val oneTwoThreeFour=oneTwo ::: threeFour
println (oneTwo + " and " + threeFour + " were not mutated.")
println ("Thus, " + oneTwoThreeFour + " is a new list")
定义了两个 List 对象 oneTwo 和 threeFour ,然后通过 ::: 操作符(其实为 ::: 方法)将两个列表连接起来。实际上由于 List 的不可修改特性,Scala 创建了一个新的 List 对象 oneTwoThreeFour 来保存两个列表连接后的值。
List 也提供了一个 :: 方法用来向 List 中添加一个元素,:: 方法(操作符)是右操作符,也就是使用 :: 右边的对象来调用它的 :: 方法,Scala 中规定所有以 : 开头的操作符都是右操作符,因此如果你自己定义以 : 开头的方法(操作符)也是右操作符。
如下面使用常量创建一个列表:
val oneTwoThree = 1 :: 2 ::3 :: Nil
println(oneTwoThree)
调用空列表对象 Nil 的 :: 方法 也就是:
val oneTwoThree = Nil.::(3).::(2).::(1)
使用元组
Scala中另外一个很有用的容器类为 Tuples ,和 List 不同的是,Tuples 可以包含不同类型的数据,而 List只能包含同类型的数据。Tuples 在方法需要返回多个结果时非常有用。(Tuple 对应到数学的 矢量 的概念)。
一旦定义了一个元组,可以使用 ._ 和 索引 来访问元组的元素(矢量的分量,注意和数组不同的是,元组的索引从1开始)。
val pair=(99,"Luftballons")
println(pair._1)
println(pair._2)
元组的实际类型取决于它的分量的类型,比如上面 pair 的类型实际为 Tuple2[Int,String],而 (‘u’,’r’,”the”,1,4,”me”) 的类型为 Tuple6[Char,Char,String,Int,Int,String] 。
目前 Scala 支持的元组的最大长度为 22 。如果有需要,你可以自己扩展更长的元组。
使用set和map
Scala Set(集合)是没有重复对象的集合,所有的元素都是唯一的。
Scala 语言的一个设计目标是让程序员可以同时利用面向对象和面向函数的方法编写代码,因此它提供的集合类分成了可以修改的集合类和不可以修改的集合类两大类型。比如 Array总是可以修改内容的,而 List 总是不可以修改内容的。类似的情况,Scala 也提供了两种 Sets 和 Maps 集合类。
比如 Scala API 定义了 Set 的 基Trait 类型 Set (Trait 的概念类似于Java中的 Interface,所不同的是Scala中的 Trait 可以有方法的实现),分两个包定义 Mutable (可变)和 Immutable (不可变),使用同样名称的子 Trait 。下图为 Trait 和类的基础关系:
使用 Set 的基本方法如下:
var jetSet = Set (“Boeing”,“Airbus”)
jetSet +=“Lear”
println(jetSet.contains(“Cessna”))
缺省情况 Set 为 Immutable Set,如果你需要使用可修改的集合类( Set 类型),你可以使用全路径来指明 Set ,比如 scala.collection.mutable.Set 。
Scala 提供的另外一个类型为 Map 类型,Map(映射)是一种可迭代的key/value结构,所有的值都可以通过键来获取。 Scala 也提供了 Mutable 和 Immutable 两种 Map 类型,如下图所示。
Map 的基本用法如下(Map 类似于其它语言中的关联数组,如 PHP )
val romanNumeral = Map ( 1 -> "I" , 2 -> "II",
3 -> "III", 4 -> "IV", 5 -> "V")
println (romanNumeral(4))
6.识别函数编程风格
Scala 语言的一个特点是支持面向函数编程,因此学习 Scala 的一个重要方面是改变之前的指令式编程思想(尤其是来自 Java 或 C# 背景的程序员),观念要向函数式编程转变。首先在看代码上要认识哪种是指令编程,哪种是函数式编程。实现这种思想上的转变,不仅仅会使你成为一个更好的 Scala 程序员,同时也会扩展你的视野,使你成为一个更好的程序员。
一个简单的原则,如果代码中含有 var 类型的变量,这段代码就是传统的指令式编程,如果代码只有 val 变量,这段代码就很有可能是函数式代码,因此学会函数式编程关键是不使用 vars 来编写代码。
来看一个简单的例子:
def printArgs ( args: Array[String]) : Unit ={
var i=0
while (i < args.length) {
println (args(i))
i+=1
}
}
来自 Java 背景的程序员开始写 Scala 代码很有可能写成上面的实现。我们试着去除 vars 变量,可以写成更符合函数式编程的代码:
def printArgs ( args: Array[String]) : Unit ={
for( arg <- args)
println(arg)
}
或者更简化为:
def printArgs ( args: Array[String]) : Unit ={
args.foreach(println)
}
这个例子也说明了尽量少用 vars 的好处,代码更简洁明了,从而也可以减少错误的发生。因此 Scala 编程的一个基本原则是,能不用 vars,尽量不用 vars,能不用 mutable 变量,尽量不用 mutable 变量,能避免函数的副作用,就尽量不产生副作用。
7.读取文件
使用脚本实现某个任务,通常需要读取文件,本节介绍 Scala 读写文件的基本方法。比如下面的例子读取文件的每行,把该行字符长度添加到行首:
import scala.io.Source
var args=Source.fromFile("/home/hadoop/test.txt") #记得在对应目录创建test.txt 文件,并写入一些内容
for( line <- args.getLines)
println(line.length + "" + line)
可以看到 Scala 引入包的方式和 Java 类似,也是通过 import 语句。文件相关的类定义在 scala.io 包中。 如果需要引入多个类, Scala 使用 _ 而非 *。