Scala基础学习 |
摘要:
在篇主要内容:如何把Scala当做工业级的便携计算器使用,如何用Scala处理数字以及其他算术操作。在这个过程中,我们将介绍一系列重要的Scala概念和惯用法。同时你还将学到作为初学者如何浏览Scaladoc文档
1. 使用Scala解释器
2. 用var和val定义变量
3. 数值类型
4. 使用操作符和函数
5. 浏览Scaladoc
Scala解释器 |
启动Scala解释器的步骤如下:
安装Scala
确保scala/bin目录位于系统PATH中
在你的操作系统中打开命令行窗口,键入scala并按Enter键
现在键人命令,然后按Enter键。每一次,解释器都会显示出结果。例如,当你键人"8*5+2"(如下面加粗的文字),将得到42:
答案被命名为res0,你可以在后续操作中使用这个名称:
正如你所看到的,解释器同时还会显示结果的类型,拿本例来说就是Int、Double和java.lang.String
还可以调用方法,根据启动解释器的方式的不同,你可能可以使用Tab键补全而不用完整地手工键人方法名。你可以试着键人res2.to,然后按Tab键,如果解释器给出了如下选项:
说明Tab键补全功能是好的。接下来键入U并再次按Tab键,你应该就能定位到单条补全如下:
同样地,可以试试按T和J方向键。在大多数实现当中,你将看到之前提交过的命令,并且可以进行编辑。用方向键和Del键将上一条命令修改为:
正如你所看到的,Scala解释器读到一个表达式,对它进行求值,将它打印出来,接着再继续读下一个表达式。这个过程被称做读取一求值一打印一循环,即REPL。从技术上讲,scala程序并不是一个解释器。实际发生的是,你输入的内容被快速地编译成字节码,然后这段字节码交由Java虚拟机执行。正因如此,大多数Scala程序员更倾向于将它称做"REPL"。
声明值和变量 |
声明值:
除了直接使用res0、res1等这些名称之外,你也可以用val定义自己的名称:
以val定义的值实际上是一个常量,你无法改变它的内容:
声明变量:
如果要声明其值可变的变量,可以用var定义其值可变的变量:
需要注意的是,不需要给出值或者变量的类型,这个信息可以从你用来初始化它的表达式推断出来。但还需注意一点,声明值或变量但不做初始化会报错。不过,在必要的时候,你也可以指定类型。例如:
由上图运行的结果可知,在Scala中,变量或函数的类型总是写在变量或函数名称的后面。这使得我们更容易阅读那些复杂类型的声明。同时还需注意的是,在变量声明或赋值语句之后,我们并没有使用分号。在Scala中,仅当同一行代码中存在多条语句时才需要用分号隔开。
常用类型 |
基本数据类型:
到目前为止你已经看到过Scala数据类型中的一些,比如Int和Double,和Java样Scala也有7种数值类型:Byte、Char、Short、Int、Long、Float和Double,以及1个Boolean类型。跟Java不同的是,这些类型是类。Scala并不刻意区分基本类型和引用类型。你可以对数字执行方法,例如:
scala> 1.toString
res6: String = 1
或者,更有意思的是,你可以:
scala> 1.to(10)
res7: scala.collection.immutable.Range.Inclusive = Range(1, 2, 3, 4, 5, 6, 7, 8,9, 10)
在Scala中,我们不需要包装类型。在基本类型和包装类型之间的转换是Scala编译器的工作。举例来说,如果你创建一个Int的数组,最终在虚拟机中得到的是一个int[]数组。
基本数据类型转换:
正如你在前面看到的那样,Scala用底层的java.lang.String类来表示字符串。不过,它通过StringOps类给字符串追加了上百种操作。举例来说,intersect方法输出两个字符串共通的一组字符:
scala> "Hello".intersect("World")
res8: String = lo
在这个表达式中java.lang.String对象"Hello"被隐式地转换成了一个StringOps对象,接着StringOps类的intersect方法被应用。因此,在使用Scala文档的时候,记得要看一下StringOps类。同样地,Scala还提供了Richlnt、RichDouble、RichChar等。它们分别提供了它们可怜的堂兄弟们Int、Double、Char等,所不具备的便捷方法。我们前面用到的to方法事实上就是Richlnt类中的方法。在表达式:1.to (10)中,Int值1首先被转换成Richlnt然后再应用to方法。
最后,还有Biglnt和BigDecimal类,用于任意大小(但有穷)的数字。这些类背后分别对应的是java.math.Biglnteger和java.math.BigDecimal
注意:在Scala中,我们用方法,而不是强制类型转换,来做数值类型之间的转换。举例来说,99.44.tolnt得到99,99.toChar得到'c'。当然和Java一样,toString将任意的对象转换成字符串。要将包含了数字的字符串转换成数值,使用tolnt或toDouble。例如,"99.44".toDouble得到99.44。
算术和操作符重载 |
Scala的算术操作符和你在Java或C++中预期的效果是一样的:
val answer=8*5+2
算术操作符:+ -*/%等操作符完成的是它们通常的工作,位操作符:&|^ >><<也一样。只有一点特别的,这些操作符实际上是方法。例如:
a+b
是如下方法调用的简写:
a.+(b)
这里的+是方法名。通常来说,你可以用:
a方法b
作为以下代码的简写:
a.方法(b)
这里的方法是一个带有两个参数的方法(一个隐式的和一个显式的)。例如
1.to (10)
可以写成:
1 to10
调用函数方法 |
除了方法之外,Scala还提供函数。相比Java,在Scala中使用数学函数更为简单,你不需要从某个类调用它的静态方法
scala> import scala.math._
import scala.math._
scala> sqrt(2)
res15: Double = 1.4142135623730951
scala> pow(2,4)
res16: Double = 16.0
scala> min(3,Pi)
res18: Double = 3.0
这些数学函数是在scala.math包中定义的。你可以用如下语句进行引入:
import scala.math._//在Scala中,_字符是"通配符",类似Java中的*字符
Scala没有静态方法,不过它有个类似的特性,叫做单例对象singleton object。通常,一个类对应有一个伴生对象companion object,其方法就跟Java中的静态方法一样
apply方法 |
在Scala中,我们通常都会使用类似函数调用的语法。举例来说,如果S是一个字符串,那么S(i)就是该字符串的第i个字符。在C++中,你会写:S[i];而在Java中,你会这样写:S.charAt(i)。在REPL中运行如下代码:
scala> "Hello"(4)
res19: Char = o
scala> "Hello"(0)
res20: Char = H
scala>
你可以把这种用法当做是()操作符的重载形式,它背后的实现原理是一个名为apply的方法。举例来说,在StringOps类的文档中,你会发现这样一个方法:
def apply(n: Int): Char
也就是说,"Hello"(4)是如下语句的简写:
"Hello".apply(4)
如果你去看Biglnt伴生对象的文档,就会看到让你将字符串或数字转换为Biglnt对象的apply方法。举例来说,如下调用
Biglnt ("1234567890")
是如下语句的简写:
Biglnt.apply("1234567890")
这个语句产出一个新的Biglnt对象,不需要使用new。例如:
Biglnt("1234567890") *Biglnt("112358111321")
像这样使用伴生对象的apply方法是Scala中构建对象的常用手法。例如,Array(l,4,9,16)返回一个数组,用的就是Array伴生对象的apply方法
Scaladoc |
Java程序员们使用Javadoc来浏览Java AP,Scala也有它自己的版本,叫做Scaladoc。相比Javadoc,浏览Scaladoc更具挑战性。Scala类通常比Java类拥有多得多的便捷方
法。有些方法使用了你还没学到的特性。而且,有些特性展示出来的是它们的实现细节而并不是使用指南。我们可以在www.scala-lang.org/api在线浏览Scaladoc,不过也可以从www.scala-lang.org/downloads#api下载一个副本安装到本地。
和Javadoc按照字母顺序列出类清单不同,Scaladoc的类清单按照包排序。如果你知道类名但不知道包名,可以用左上角的过滤器,参照下图:
点击×号可以清空过滤器。注意每个类名旁边的O和C,它们分别链接到对应的类(C)或伴生对象(O)。由于信息量大,Scaladoc可能读起来会比较累人。请记住以下这些小窍门:
■ 如果你想使用数值类型,记得看看Richlnt、RichDouble等。同理,如果想使用字符串,记得看看SpringOps
■ 那些数学函数位于scala.math包中,而不是位于某个类中
■ 有时你会看到名称比较奇怪的函数。比如,Biglnt有一个方法叫做unary_-。这就是你定义前置的负操作符-x的方式
■ 标记为implicit的方法对应的是自动(隐式)转换。比如,Biglnt对象拥有在需要时自动被调用的由int和long转换为Biglnt的方法
■ 方法可以以函数作为参数。例如,StringOps的count方法需要传人一个接受单个Char并返回true或false的函数,用于指定哪些字符应当被清点:
def count(p: (Char)=>Boolean) : Int
调用类似方法时,你通常可以一种非常紧凑的表示法给出函数定义。例如,s.count(_.isUpper)的作用是清点所有大写字母的数量
■ 你时不时地会遇到类似Range或Seq[Char]这样的类。它们的含义和你的直觉告诉你的一样:一个是数字区间,一个是字符序列
如果,您认为阅读这篇博客让您有些收获,不妨点击一下右下角的【推荐】。
如果,您希望更容易地发现我的新博客,不妨点击一下左下角的【关注我】。
如果,您对我的博客所讲述的内容有兴趣,请继续关注我的后续博客,我是【Sunddenly】。本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。