9
23
2013
0

[转]《闪灵》:理想主义的丧钟

http://movie.douban.com/review/6293444/

  《闪灵》的魅力,很大程度上在于它的意义空白与不确定性。从来都是没逻辑(或至少是看不出逻辑的东西)才最恐怖,而《闪灵》恰恰在一个成逻辑的体系上 加上许多没逻辑的细节,它们与主体的关系若即若离,背后的意味则令人不寒而栗,从而成就一个意蕴丰富且耐人寻味的文本——于是几十年来的人们总结出了种种 解读:山庄是男权社会的缩影、预示着美国梦的破裂、事件其实是在影射白人对印第安人的惨绝人寰大屠杀……这些说法均非常有道理,不过未免过于宏观了,如果 编剧从一开始就想这么多,后续工作是很难进行下去的。窃以为,解读它的关键还是在于牢牢抓住影片中真正成逻辑的故事、即杰克尼克尔森由慈父演变为狂魔的悲 剧,其余元素皆是以它为基础的,编剧所做的更多是把握住了作家梦碎、种族屠杀、男权倒塌几大事件之间的共性。
  
  
  诚然,《闪灵》的母题根植于在民权(黑人)和女权(妻子)运动下摇摇欲坠的白人男权(这个主题在当时许多恐怖片中都有所体现,我曾写过一篇讲《驱魔人》的:http://movie.douban.com/review/5890748/), 不过值得注意的是,山庄中的两代杀人案,都是由隶属于社会底层的看管员犯下的,换言之如果他们是山庄的所有者(即成功人士),这个故事也便不再成立了。 《闪灵》与斯蒂芬金的许多其他作品一样,带有强烈的自传色彩,是一部不折不扣的宅男与丧男文学;如果说影片就这一层面的发掘还不够震撼的话,那么原著中那 连篇累牍的心理描写,道尽了酒后幻觉的狰狞与灵感枯竭的颓唐,满纸悲凉令人不忍回望。杰克的悲剧从影片开始处就已经注定了,因为他是一个在理想和现实的巨 大鸿沟间痛苦挣扎的失败者。是人就做梦,偏偏他的梦是最二逼的那种,比整天幻想往夜店里一站就被大老板包养还要二逼上很多——他想成为一个作家,并身体力 行地在这条路上跋涉了许多年,终于到了这个拖家带口却两袖清风的地步。
  
  
  
  不仅现实本身给予了他无穷的压力,妻子的贤惠亦令他无比的痛苦。她起初想必也是他的书迷(当被告知山庄中曾发生过恶性命案时,杰克说“我妻子 不会在意的,她最喜欢读恐怖故事”),结婚之前大概也同他有过极为浪漫的时光(杰克在楼梯上发飙时曾说过一句“light of my life”),至今仍在坚定地支持他的写作,只不过起了一点变化:孩子降生以后,她的柴米油盐酱醋茶程度要远甚于以往了,儿子而非丈夫占据了她生活的中 心。这一方面将这对夫妇划分到了“理想”与“现实”两个平行世界——永远埋首家务的妻子,一举一动都仿佛在暗示着他这个世界有多么的世俗;另一方面,纵使 琐事缠身、境况贫穷,妻子也始终满面笑容,这就没有给杰克任何发泄的借口,只能把种种郁结堆积在心灵深处。
  
  
  
  至于儿子,杰克显然是很爱他的。他之所以会伤害儿子,是因为丹尼把他的卷子扔得到处都是,即严重影响了他的工作。教书本身就是他赖以谋生的辅 职,主职不顺利,辅职还与儿子一起牵扯他的精力,压力之大可想而知,做出过激举动也不是不能理解——养育后代着实是一件很考验责任心和人格成熟度的事情, 其实很多人在为人父母之初乃至为人父母很多年后,仍会难免产生“后代是负累,自己不配为人父”的想法。杰克曾向“酒保”抱怨“只要我活着,她就一天不让我 忘记那件事”,可实际上妻子对杰克误伤儿子一事还是比较通情达理的,故真正令杰克烦躁的是他内心的负疚与焦虑:步入而立之年,事业上却毫无建树,连妻儿都 无法养活,沦落到要来旅馆看门的地步,拖地洗碗养儿子的妻子,成了这个家庭实际上的顶梁柱。
  
  
  
  更令他不堪重负的是她无条件的包容——在影片第43分钟左右,杰克坐在打字机旁陷入灵感枯竭的痛苦,胡乱敲些见不得人的字句,不想妻子忽然出 现,问候一番后来了一句“也许你能给我看看你又写了些什么”,导致一直努力压抑怒火的杰克瞬间爆炸。或许不少人都曾有这种体会:在你极度沮丧、极度自我怀 疑的时候,别人越说你牛逼,你就越觉得对方是在蒙骗你、怜悯你、伤你的自尊;但如果说你傻逼呢?你会想揍他一顿。很多时候躁郁的根源往往都不是外部压力, 而是自高自大的同时自暴自弃,由于和直面挑战相比自暴自弃要来得更容易,所以我们宁可做一个总是处于躁郁状态的大傻逼。
  
  
  
  因此杰克的怒火是一种迁怒:他觉得自己的理想被理想的家庭毁灭了,对妻儿的爱正撕扯着他对写作的爱,理想主义者那颗脆弱的理想主义心灵,在严 苛的现实面前四分五裂得极其彻底。是放下这份根本看不到半分前景的追求、履行对家庭的责任,抑或是假装忘记自己身为男人的尊严,念叨着“凯撒一旦越过卢比 孔河便不再回头”,厚颜无耻地继续跋涉于这毁灭之路?梅尔维尔也是在死后很多年才得以千古。在第59分钟左右,杰克梦到自己“亲手杀了妻儿,并将他们的尸 体切成一片一片的”,这个段落反应了他的真正愿望:毁掉家庭、回到没有负累的时代,那时他年轻而富有活力,总能恃才傲物、愤世嫉俗,能够自由地追逐理想, 天真地以为一笔在手,未来一片坦途。
  
  
  
  不过,杰克的处境,还远远未达到一定要以灭妻杀子来解脱的地步,真正打垮他的其实是自身的大男子主义。杰克,或者说斯蒂芬金,所追求的都不是 纯粹的文学,而是在保有风骨的同时,借助文学财源广进、功成名就,所以从酒店宴会一幕中可以看出,他的奢望是非常直男式的:美酒、豪宅、美女与排场。现实 呢?只有空空如也的钱包和寒酸廉价的夹克。如果他能放下这对他来说意义不大的大男子傲慢(当然这很难做到),以纯粹的心态专注于写作、珍重他所拥有的一 切,一样可以经营起一个清贫却幸福的小家庭,不管能否收获成功,结局总要好过埋尸雪地。
  
  
  
  山庄之所以能对杰克产生这么大的影响,是因为它释放出了杰克浮躁的幻想,令他沉沦于虚幻的欲望,不愿再回到妻儿所处的“现实”之中。杀妻儿的 前任看管员对杰克所说的最后一句话为:“you’ve always been the caretaker.”你一直都是这里的看门人!意即你就是我,我就是你,这个酒店是一个恐怖而贪婪的幻境,禁锢着无数战胜不了自身纠结、宁可沉沦逃避的 卑微灵魂。所以杰克最终的结局,就是被钉在那张标有“1921”的黑白照片上,身着华贵西服,面露志得意满的笑容,处在宴会人群的中央,在虚幻的时空中成 就一场爵士时代的美国梦——比起在山下敲打字机、收退稿信、忍受心灵的煎熬,他更喜欢这里“功成名就”。
  
  
  
  因此《闪灵》的真正恐怖之处,在于你7岁听见名号,14岁看个热闹,21岁勉强读出门道,28岁你逐步朝其中靠近,待到35岁,你恐怕已经置 身于漩涡中心——杰克的挣扎就是你的挣扎,人人都处在一样的悲剧里,毕竟一个出身平凡的人所能拥有的完全属于自己的时间是很少的,充其量是十几岁到二十岁 这几年而已,若浪费掉了,就再也无法回头。很快你就要去低三下四地装孙子,花很多时间养孩子,变成自己最讨厌的那个人,把曾经迷恋过的一切都忘记,斗志一 丧千里,光阴白驹过隙,写字时手会抖个不停,看过的书立刻抛到脑后,最后意识到自己的理想都被别人实现了,满怀郁郁地落进棺材里。为了做一个正常的“社会 人”,你不得不把你一多半的生命分给你的孩子,不巧在他学会承担责任之前,他会固执地恨你十年,认为你没有担当、勇气和能力,像你当初一样,沉浸在“残酷 青春”的幻觉里。其实什么不比青春残酷呢?幼年的残酷无法表述,老年的残酷没有尽头,中年的残酷是一片荒芜的黑洞,谁人都挣脱不出。
  
  
  
  生命是一条不归路,越往前走越凄凉,终点是死亡。你是流连于那座幻境重重的宾馆、让自己被钉死在永恒的痛苦轮回上呢,还是去珍重那些真正关心 你、支持你、喜欢你的人,然后抛开那些无意义的骄傲,勇敢地走下山去呢?从道理上,我们都知道后者是对的;可在现实中,却总是选了前者,因为逃避来得更容 易。原著中的杰克、或说斯蒂芬金最终成功走下了山重获新生,然而还有更多或许比他更有才华的人却还攀爬在通往山顶的道路上。
  
  
  
  THE END

Category: 未分类 | Tags: 闪灵 The Shining
6
14
2013
23

随手记

在遥远北方那被称作斯维茨约的土地上,有一块石头。它有一百英里高,一百英里宽。每一千年,会有一只小鸟来到这里,用它磨尖自己的喙。当这块石头如此消磨殆尽的时候,永恒便又度过了一天。

——Hendrik Willem Van Loon

Category: 其他 | Tags:
4
20
2013
1

Scala学习笔记-Part.02

样本类和模式匹配

样本类(case class)和模式匹配(pattern matching)。

简单的例子

假设要建立一个操作数学表达式的库,就要先定义输入的数据。为了简单,现在只关注由变量、数字、一元及二元操作符组成的数学表达式上:

样本类

  abstract class Expr
  case class Var(name: String) extends Expr
  case class Number(num: Double) extends Expr
  case class UnOp(operator: String, arg: Expr) extends Expr
  case class BinOp(operator: String,
      left: Expr, right: Expr) extends Expr

上面为表达式定义了一个抽象的基类,四个子类分别代表四种具体的表达式。要注意的是每个子类都有一个case修饰符,会被编译器识别为样本类。

样本类在使用工厂方法创建时可以省掉new

val v = Var("x")

这个特点让方法嵌套时少写new让代码看起来更加简洁:

val op = binOp("+", Number(1), v)

样本类的另一个特点是参数列表中所有的参数隐式获得了val前缀,被作为字段维护:

scala> v.name
res0: String = x

scala> op.left
res1: Expr = Number(1.0)

编译器为样本类添加了可读性更强的toStringhashCodeequals方法:

  scala> println(op)
  BinOp(+,Number(1.0),Var(x))

  scala> op.right == Var("x")
  res3: Boolean = true

先来看一下格式。在格式上相当于把Java的switch格式:

  switch (selector) { alternatives }

中括号里的选择器移到了match关键字的前面:

  selector match { alternatives }

下面了一些基本的数学简化操作。下面的操作不用计算,因为值是不会变的:

  UnOp("-", UnOp("-", e)) => e // 负负得正
  BinOp("+", e, Number(0)) => e // 加0
  BinOp("*", e, Number(1)) => e // 乘1

定义一个simplifyTop来简化运算:

  def simplifyTop(expr: Expr): Expr = expr match {
    case UnOp("-", UnOp("-", e)) => e // Double negation
    case BinOp("+", e, Number(0)) => e // Adding zero
    case BinOp("*", e, Number(1)) => e // Multiplying by one
    case _ => expr
  }

方法simplifyTop接收一个Expr类型的参数。这里参数expr作为选择器匹配各个备选项,_为通配模式能匹配所有的值,相当于Java中的default。箭头=>分开的模式与表达式。 调用:

  scala> simplifyTop(UnOp("-", UnOp("-", Var("x"))))
  res4: Expr = Var(x)
  • match是一种表达式,所以有返回结果。
  • 一个case不会走到下一个case。
  • 如果一项也没有匹配成功,会抛出MatchError异常。如果不想要异常要么把所有可能性都写上;要么加一个_的默认情况。
  expr match {
    case BinOp(op, left, right) =>
      println(expr +" is a binary operation")
    case _ =>
  }

这个表达式在两种情况下都会返回Unit值'()'。

模式的种类

通配模式

通配模式“_”匹配所有的结果:

  expr match {
    case BinOp(op, left, right) =>
      println(expr +"is a binary operation")
    case _ =>
  }

通配符还可以省略省略不用关注的内容,比如函数的参数名:

  expr match {
    case BinOp(_, _, _) => println(expr +"is a binary operation")
    case _ => println("It's something else")
  }

常量模式

任何字面量都可以用作常量,如nil5true"hello"

  def describe(x: Any) = x match {
    case 5 => "five"
    case true => "truth"
    case "hello" => "hi!"
    case Nil => "the empty list"
    case _ => "something else"
  }

效果:

  scala> describe(5)
  res5: java.lang.String = five

  scala> describe(true)
  res6: java.lang.String = truth

  scala> describe("hello")
  res7: java.lang.String = hi!

  scala> describe(Nil)
  res8: java.lang.String = the empty list

  scala> describe(List(1,2,3))
  res9: java.lang.String = something else

变量模式

变量类似通配模式,只不过有个变量名所以可以在后面的表达式中操作这个变量:

  expr match {
    case 0 => "zero"
    case somethingElse => "not zero: "+ somethingElse
  }

变量模式与常量模式的区别

常量不止有字面形式,还有用符号名的(比如Nil)。这样看起来就很容易与变量模式搞混:

  scala> import Math.{E, Pi}
  import Math.{E, Pi}

  scala> E match {
       | case Pi => "strange math? Pi = "+ Pi
       | case _ => "OK"
       | }
  res10: java.lang.String = OK

上面的EPi都是常量。对Scala编译器来说小写字母开头都作为变量,其他引用被认为是常量。下面的例子中想建立一个小写的pi就匹配到常量Pi了:

  scala> val pi = Math.Pi
  pi: Double = 3.141592653589793

  scala> E match {
       | case pi => "strange math? Pi = "+ pi
       | }
  res11: java.lang.String = strange math? Pi = 2.7182818...

在这个变量模式情况下,不能使用通配模式。因为变量模式已经可以匹配所有情况了:

  scala> E match {
       | case pi => "strange math? Pi = "+ pi
       | case _ => "OK"
       | }
  <console>:9: error: unreachable code
           case _ => "OK"
                     ^

其实也有强制使用小写常量名的方式:this.piobj.pi的形式表示是常量模式;如果这样还没有用,可以用反引号包起来,如:

  scala> E match {
       | case `pi` => "strange math? Pi = "+ pi
       | case _ => "OK"
       | }
  res13: java.lang.String = OK

反引号也可以用来处理其他的编码问题,如对于标识符来说,因为yield是Scala的保留字 所以不能写Thread.yield(),但可以写成:

Thread.`yield`()

构造器模式

这个模式是真正牛X的模式,不仅检查对象是否是样本类的成员,还检查对象的构造器参数是否符合指定模式。

Scala的模式支持深度匹配(deep match)。不止检查对象是否一致而且还检查对象的内容是否匹配内层模式。由于额外的模式自身可以形成构造器模式,因此可以检查到对象内部的任意深度。

如下面的代码不仅检查了顶层的对象是BinOp,而且第三个构造参数是Number,而且它的值为0

  expr match {
    case BinOp("+", e, Number(0)) => println("a deep match")
    case _ =>
  }

序列模式

指定匹配序列中任意元素,如指定开始为0:

  expr match {
    case List(0, _, _) => println("found it")
    case _ =>
  }

不固定长度用_*

  expr match {
    case List(0, _*) => println("found it")
    case _ =>
  }

元组模式

  def tupleDemo(expr: Any) =
    expr match {
      case (a, b, c) => println("matched "+ a + b + c)
      case _ =>
    }

调用:

  scala> tupleDemo(("a ", 3, "-tuple"))
  matched a 3-tuple

类型模式

可以当成类型测试和类型转换的简易替代:

  def generalSize(x: Any) = x match {
    case s: String => s.length
    case m: Map[_, _] => m.size
    case _ => -1
  }

调用的例子:

  scala> generalSize("abc")
  res14: Int = 3

  scala> generalSize(Map(1 -> 'a', 2 -> 'b'))
  res15: Int = 2

  scala> generalSize(Math.Pi)
  res16: Int = -1

注意方法中sx虽然都指向同一个对象,但一个类型是String一个类型是Any。所以可以写成s.length不可以写成x.length

另一个测试类型的方法:

expr.isInstanceOf[String]

另一个转换类型的方法:

expr.asInstanceOf[String]

使用类型转换的例子:

  if (x.isInstanceOf[String]) {
    val s = x.asInstanceOf[String]
    s.length
  } else ...

类型擦除

和Java一样,对于除了数组以外其他集合都采用了泛型擦除(erasure)。就是在运行时不知道集合泛型类型。

如对于Map[Int,Int],到了运行时就不知道两个类型是什么类型了。所以对于泛型的模式匹配,编译器会有警告信息:

  scala> def isIntIntMap(x: Any) = x match {
       | case m: Map[Int, Int] => true
       | case _ => false
       | }
  warning: there were unchecked warnings; re-run with
     -unchecked for details
  isIntIntMap: (Any)Boolean

在启动编译器时加上检查开关可以看到更多详细信息:

  scala> :quit
  $ scala -unchecked
  Welcome to Scala version 2.7.2
  (Java HotSpot(TM) Client VM, Java 1.5.0_13).
  Type in expressions to have them evaluated.
  Type :help for more information.

scala> def isIntIntMap(x: Any) = x match {
     | case m: Map[Int, Int] => true
     | case _ => false
     | }
  <console>:5: warning: non variable type-argument Int in
  type pattern is unchecked since it is eliminated by erasure
           case m: Map[Int, Int] => true
                   ^

所以对于不同的类型,上面函数结果都是true

  scala> isIntIntMap(Map(1 -> 1))
  res17: Boolean = true

  scala> isIntIntMap(Map("abc" -> "abc"))
  res18: Boolean = true

数组和Java一样,是没有类型擦除的:

  scala> def isStringArray(x: Any) = x match {
       | case a: Array[String] => "yes"
       | case _ => "no"
       | }
  isStringArray: (Any)java.lang.String

  scala> val as = Array("abc")
  as: Array[java.lang.String] = Array(abc)

  scala> isStringArray(as)
  res19: java.lang.String = yes

  scala> val ai = Array(1, 2, 3)
  ai: Array[Int] = Array(1, 2, 3)

  scala> isStringArray(ai)
  res20: java.lang.String = no

变量绑定

除了独立的变量模式外,还可以对任何其他模式添加变量。 作用时在匹配成功后,变量就是匹配成功的对象了。 方式为写上变量名、一个@符号和模式。

比如要匹配abs出现了两次的地方(做了两次绝对值计算等于没有算):

  expr match {
    case UnOp("abs", e @ UnOp("abs", _)) => e
    case _ =>
  }

守卫模式

Scala要求模式是线性的,即模式变量只能在模式中出现一次。

如,想要把e+e这个重复加法替换成乘法e*2

  BinOp("+", Var("x"), Var("x"))

等于:

  BinOp("*", Var("x"), Number(2))

用下面的方式x重复出现了,所以有问题:

  scala> def simplifyAdd(e: Expr) = e match {
       | case BinOp("+", x, x) => BinOp("*", x, Number(2))
       | case _ => e
       | }
  <console>:10: error: x is already defined as value x
           case BinOp("+", x, x) => BinOp("*", x, Number(2))
                              ^

守卫模式(pattern guard)很像for循环中的if过滤条件。接在匹配模式后面的、用if开始的、使用模式中变量的表达式:

  scala> def simplifyAdd(e: Expr) = e match {
       | case BinOp("+", x, y) if x == y =>
       | BinOp("*", x, Number(2))
       | case _ => e
       | }
  simplifyAdd: (Expr)Expr

其他的例子,如只匹配正整数和只匹配以a开始的字符串:

  // match only positive integers
  case n: Int if 0 < n => ...

  // match only strings starting with the letter `a'
  case s: String if s(0) == 'a' => ...

模式重叠

  def simplifyAll(expr: Expr): Expr = expr match {
    case UnOp("-", UnOp("-", e)) =>
      simplifyAll(e) // `-' is its own inverse
    case BinOp("+", e, Number(0)) =>
      simplifyAll(e) // `0' is a neutral element for `+'
    case BinOp("*", e, Number(1)) =>
      simplifyAll(e) // `1' is a neutral element for `*'
    case UnOp(op, e) =>
      UnOp(op, simplifyAll(e))
    case BinOp(op, l, r) =>
      BinOp(op, simplifyAll(l), simplifyAll(r))
    case _ => expr
  }

注意这个方法的第四个和第五个匹配样本的参数都是变量,而且对应的操作采用递归。因为四和五的匹配范围比前三个更加广,所以建立放在后面。如果放在前面的话会有警告。

如下面的第一个样本能匹配任何第二个样本能匹配的情况:

  scala> def simplifyBad(expr: Expr): Expr = expr match {
       | case UnOp(op, e) => UnOp(op, simplifyBad(e))
       | case UnOp("-", UnOp("-", e)) => e
       | }
  <console>:17: error: unreachable code
           case UnOp("-", UnOp("-", e)) => e
                                           ^

封闭类

前面说过Scala里如果所有的样本都没有匹配,那是会抛异常的。为了全都匹配,程序员会给匹配加上一个默认匹配项处理默认情况。

实际上Scala编译器已经可以检测match表达式中遗漏的情况,但新的样本类可以定义在任何地方。比如我们的Expr有四个样本类,但可以在别的编译单元中再写其他的实现类。

所以有一个方案:让样本类的超类被封闭(sealed),这样就不能在别的文件中添加新的子类。格式只要加一个sealed关键字:

  sealed abstract class Expr
  case class Var(name: String) extends Expr
  case class Number(num: Double) extends Expr
  case class UnOp(operator: String, arg: Expr) extends Expr
  case class BinOp(operator: String,
      left: Expr, right: Expr) extends Expr

如果代码里漏掉可能的模式:

  def describe(e: Expr): String = e match {
    case Number(_) => "a number"
    case Var(_) => "a variable"
  }

编译器会警告UnOpBinOp没有处理:

  warning: match is not exhaustive!
  missing combination UnOp
  missing combination BinOp

如果程序员确实知道这两种情况不可能发生,就是要在这两种情况下抛异常。可以手动加上让编译器闭嘴:

  def describe(e: Expr): String = e match {
    case Number(_) => "a number"
    case Var(_) => "a variable"
    case _ => throw new RuntimeException // Should not happen
  }

还有一个方法是对变量e添加注释@unchecked

  def describe(e: Expr): String = (e: @unchecked) match {
    case Number(_) => "a number"
    case Var(_) => "a variable"
  }


Option类型

非必填类型:可以存值,也可以为None对象代表没有值。形式为Some(x)x表示非必要。比如Scala的Map类型:

  scala> val capitals =
       | Map("France" -> "Paris", "Japan" -> "Tokyo")
  capitals:
    scala.collection.immutable.Map[java.lang.String,
    java.lang.String] = Map(France -> Paris, Japan -> Tokyo)

  scala> capitals get "France"
  res21: Option[java.lang.String] = Some(Paris)

  scala> capitals get "North Pole"
  res22: Option[java.lang.String] = None

应用模式匹配处理有值和没有值的情况:

  scala> def show(x: Option[String]) = x match {
       | case Some(s) => s
       | case None => "?"
       | }
  show: (Option[String])String

  scala> show(capitals get "Japan")
  res23: String = Tokyo

  scala> show(capitals get "France")
  res24: String = Paris

  scala> show(capitals get "North Pole")
  res25: String = ?

在Java里Map没有值时返回的是null,如果忘记检查会引起空指针异常。而在Scala里对于一个Map[Int,Int]是不可能返回null的。

使用Option类型的优点在于:

  • Option[String]从字面上看就已经提醒了程序员内容可能为None
  • 在Java中如果变量为空要到运行时才抛出空指针异常,而Scala中Option类型让编译器就已经提供了检查:编译器会在把Option[String]当作String使用时报错。

模式无处不在

变量定义

通过类型定义变量:

  scala> val myTuple = (123, "abc")
  myTuple: (Int, java.lang.String) = (123,abc)

用模式匹配代替类型声明:

  scala> val (number, string) = myTuple
  number: Int = 123
  string: java.lang.String = abc

这种方式用在指定精确类型的样本类时用得比较多:

  scala> val exp = new BinOp("*", Number(5), Number(1))
  exp: BinOp = BinOp(*,Number(5.0),Number(1.0))

  scala> val BinOp(op, left, right) = exp
  op: String = *
  left: Expr = Number(5.0)
  right: Expr = Number(1.0)

偏函数的样本序列

花括号case对应的样本本来就是函数字面量,可以用在任何用函数字面量的地方。而且还是有相当多个可选的函数字面量。如:

  val withDefault: Option[Int] => Int = {
    case Some(x) => x
    case None => 0
  }

调用:

  scala> withDefault(Some(10))
  res25: Int = 10

  scala> withDefault(None)
  res26: Int = 0

这样的方式很适合Actor应用:

  react {
    case (name: String, actor: Actor) => {
      actor ! getip(name)
      act()
    }
    case msg => {
      println("Unhandled message: "+ msg)
      act()
    }
  }

顺带提一下应用在偏(partial)函数上的应用。如果是不支持的值上会产生一个运行时异常。

如,下面的偏函数能返回整数列表的第二个元素:

  val second: List[Int] => Int = {
    case x :: y :: _ => y
  }

编译器会提示匹配不全:

  <console>:17: warning: match is not exhaustive!
  missing combination Nil

如果传递给它一个三元素列表,它的执行没有问题。但是一个空列表就不行了:

  scala> second(List(5,6,7))
  res24: Int = 6

  scala> second(List())
  scala.MatchError: List()
   at $anonfun$1.apply(<console>:17)
   at $anonfun$1.apply(<console>:17)

如果要检查一个偏函数是否有定义,一定要告诉编译器正在使用的函数是偏函数。类型Lint[Int] => Int包含了不管是否是偏函数的,从整数列表到整数的所有函数。仅包含整数列表到的偏函数的,应该写成`Partialfunction[List[Int],Int]。

下面是偏函数的定义例子:

  val second: PartialFunction[List[Int],Int] = {
    case x :: y :: _ => y
  }

偏函数有一个idDefineAt方法来测试函数对某个值是否有定义。以这个例子来说,对于至少两个元素的列表是有定义的:

  scala> second.isDefinedAt(List(5,6,7))
  res27: Boolean = true

  scala> second.isDefinedAt(List())
  res28: Boolean = false

Scala在编译器在把这样的表达式转为偏函数时会对模式进行两次翻译:一次是真实函数的实现;另一次是测试函数是否对参数有定义的实现。

例如上面的函数宣布量`{case x :: y
_ => y`会被翻译成:
  new PartialFunction[List[Int], Int] {
    def apply(xs: List[Int]) = xs match {
      case x :: y :: _ => y
    }
    def isDefinedAt(xs: List[Int]) = xs match {
      case x :: y :: _ => true
      case _ => false
    }
  }

这只有在声明类型为PartialFunction时才会发生。如果只是Function1或没有声明,函数字面量会编译为完整的函数。

偏函数可能会引起运行时的异常,所以在调用前用isDefineAt检查一下。

for表达式

典型例子,每个元素都是(country,city)

  scala> for ((country, city) <- capitals)
       | println("The capital of "+ country +" is "+ city)
  The capital of France is Paris
  The capital of Japan is Tokyo

当然也有元素不匹配模式的情况,下面例子中不匹配的会被丢弃:

  scala> val results = List(Some("apple"), None,
       | Some("orange"))
  results: List[Option[java.lang.String]] = List(Some(apple),
      None, Some(orange))

  scala> for (Some(fruit) <- results) println(fruit)
  apple
  orange

大型的例子

目标是生成公式((a / (b * c) + 1 / n) / 3)显示形式为:

  a 1
----- + -
b * c n
---------
    3

先来看:

  BinOp("+",
        BinOp("*",
              BinOp("+", Var("x"), Var("y")),
              Var("z")),
        Number(1))

应该输出(x+y)*z+1(x+y)是有括号的,但是最外层不要括号。所以要先解决优先级问题:

  Map(
    "|" -> 0, "||" -> 0,
    "&" -> 1, "&&" -> 1, ...
  )

当然还有改进的空间,更好的方法是只定义递减的优先级操作符。然后根据它来计算:

    // Contains operators in groups of increasing precedence
    private val opGroups =
      Array(
        Set("|", "||"),
        Set("&", "&&"),
        Set("^"),
        Set("==", "!="),
        Set("<", "<=", ">", ">="),
        Set("+", "-"),
        Set("*", "%")
      )

再定义一个操作符与优先级映射的变量precedence,映射的内容是通过处理上面定义的层级生成的:

    // A mapping from operators to their precedence
    private val precedence = {
      val assocs =
        for {
          i <- 0 until opGroups.length
          op <- opGroups(i)
        } yield op -> i
      Map() ++ assocs
    }

    private val unaryPrecedence = opGroups.length
    private val fractionPrecedence = -1

如:

BinOp("-", Var("a"), BinOp("-", Var("b"), Var("c")))

将被处理为a - (b - c)。如果当前操作符优先级小于外部操作符的优先级,那oper就要被放在括号里,不然按原样返回。

具体实现:

      case BinOp(op, left, right) =>
        val opPrec = precedence(op)
        val l = format(left, opPrec)
        val r = format(right, opPrec + 1)
        val oper = l beside elem(" "+ op +" ") beside r
        if (enclPrec <= opPrec) oper
        else elem("(") beside oper beside elem(")")

最后再给一个公开的方法,不用优先级参数就可以格式化公式:

    def format(e: Expr): Element = format(e, 0)

全部代码:

//compile this along with ../compo-inherit/LayoutElement.scala

  package org.stairwaybook.expr
  import layout.Element.elem
  
  sealed abstract class Expr
  case class Var(name: String) extends Expr
  case class Number(num: Double) extends Expr
  case class UnOp(operator: String, arg: Expr) extends Expr
  case class BinOp(operator: String,
      left: Expr, right: Expr) extends Expr
  
  class ExprFormatter {
  
    // Contains operators in groups of increasing precedence
    private val opGroups =
      Array(
        Set("|", "||"),
        Set("&", "&&"),
        Set("^"),
        Set("==", "!="),
        Set("<", "<=", ">", ">="),
        Set("+", "-"),
        Set("*", "%")
      )
  
    // A mapping from operators to their precedence
    private val precedence = {
      val assocs =
        for {
          i <- 0 until opGroups.length
          op <- opGroups(i)
        } yield op -> i
      Map() ++ assocs
    }
  
    private val unaryPrecedence = opGroups.length
    private val fractionPrecedence = -1
  
    // continued in Listing 15.21...

  import org.stairwaybook.layout.Element

  // ...continued from Listing 15.20
  
  private def format(e: Expr, enclPrec: Int): Element =
  
    e match {
  
      case Var(name) =>
        elem(name)
  
      case Number(num) =>
        def stripDot(s: String) =
          if (s endsWith ".0") s.substring(0, s.length - 2)
          else s
        elem(stripDot(num.toString))
  
      case UnOp(op, arg) =>
        elem(op) beside format(arg, unaryPrecedence)
  
      case BinOp("/", left, right) =>
        val top = format(left, fractionPrecedence)
        val bot = format(right, fractionPrecedence)
        val line = elem('-', top.width max bot.width, 1)
        val frac = top above line above bot
        if (enclPrec != fractionPrecedence) frac
        else elem(" ") beside frac beside elem(" ")
  
      case BinOp(op, left, right) =>
        val opPrec = precedence(op)
        val l = format(left, opPrec)
        val r = format(right, opPrec + 1)
        val oper = l beside elem(" "+ op +" ") beside r
        if (enclPrec <= opPrec) oper
        else elem("(") beside oper beside elem(")")
    }
  
    def format(e: Expr): Element = format(e, 0)
  }

具体调用的例子:

  import org.stairwaybook.expr._

  object Express extends Application {

    val f = new ExprFormatter

    val e1 = BinOp("*", BinOp("/", Number(1), Number(2)),
                        BinOp("+", Var("x"), Number(1)))
    val e2 = BinOp("+", BinOp("/", Var("x"), Number(2)),
                        BinOp("/", Number(1.5), Var("x")))
    val e3 = BinOp("/", e1, e2)

    def show(e: Expr) = println(f.format(e)+ "\n\n")

    for (val e <- Array(e1, e2, e3)) show(e)
}
  scala Express

下一个问题是格式化方法的实现。定义一个format方法,参数为表达式类型的e: Expr和操作符的优先级enclPrec: Int(如果没有这个操作符,那优先级就应该是0)。

注意format是私有方法,完成大部分工作。最后一个公开的同名方法format提供入口。内部还有一个stripDot方法来去掉如2.0.0部分。

  private def format(e: Expr, enclPrec: Int): Element =

    e match {

      case Var(name) =>
        elem(name)

      case Number(num) =>
        def stripDot(s: String) =
          if (s endsWith ".0") s.substring(0, s.length - 2)
          else s
        elem(stripDot(num.toString))

      case UnOp(op, arg) =>
        elem(op) beside format(arg, unaryPrecedence)

      case BinOp("/", left, right) =>
        val top = format(left, fractionPrecedence)
        val bot = format(right, fractionPrecedence)
        val line = elem('-', top.width max bot.width, 1)
        val frac = top above line above bot
        if (enclPrec != fractionPrecedence) frac
        else elem(" ") beside frac beside elem(" ")

      case BinOp(op, left, right) =>
        val opPrec = precedence(op)
        val l = format(left, opPrec)
        val r = format(right, opPrec + 1)
        val oper = l beside elem(" "+ op +" ") beside r
        if (enclPrec <= opPrec) oper
        else elem("(") beside oper beside elem(")")
    }

    def format(e: Expr): Element = format(e, 0)
  }

分析具体的逻辑:

如果是变量,结果就是变量名。

  case Var(name) =>
    elem(name)

如果是数字,结果是格式化后的数字,如2.0格式化为2

      case Number(num) =>
        def stripDot(s: String) =
          if (s endsWith ".0") s.substring(0, s.length - 2)
          else s
        elem(stripDot(num.toString))

对于一元操作符的处理结果为操作op和最高环境优先级格式化参数arg的结果组成。这样如果arg是除了分数以外的二元操作就不会出现在括号中。

      case UnOp(op, arg) =>
        elem(op) beside format(arg, unaryPrecedence)

如果是分数,则按上下位置放置。但仅仅上下的位置还不够。因为这样公不清主次:

a
-
b
-
c

有必要强化层次:

 a
 -
 b
---
 c

实现的代码这个样子的:

      case BinOp("/", left, right) =>
        val top = format(left, fractionPrecedence)
        val bot = format(right, fractionPrecedence)
        val line = elem('-', top.width max bot.width, 1)
        val frac = top above line above bot
        if (enclPrec != fractionPrecedence) frac
        else elem(" ") beside frac beside elem(" ")

然后了除法以外的其他二元操作符。在这里要注意一下优先级问题:

左操作数的优先级是操作符opopPrec,而右操作数的优先级要再加1。这样保证了括号也同样反映正确的优先级。

使用列表

列表字面量

再简单回顾一下:

  val fruit = List("apples", "oranges", "pears")
  val nums = List(1, 2, 3, 4)
  val diag3 =
    List(
      List(1, 0, 0),
      List(0, 1, 0),
      List(0, 0, 1)
    )
  val empty = List()"brush: scala"

注意列表是不呆变的。

列表类型

列表是同质化的(homogeneous),所有的成员都有相同的类型,中括号描述成员类型List[T]

val fruit: List[String] = List("apples", "oranges", "pears")
  val nums: List[Int] = List(1, 2, 3, 4)
  val diag3: List[List[Int]] =
    List(
      List(1, 0, 0),
      List(0, 1, 0),
      List(0, 0, 1)
    )
  val empty: List[Nothing] = List()

Scala里的列表类是协变的(covariant)。如果ST的子类,那List[S]也是List[T]的子类。

由于Nothing是所有类的子类,所以List[Nothing]是所有List[T]类型的子类:

  // List() is also of type List[String]!
  val xs: List[String] = List()

构造列表

Nil代表空列表;::(发音为“cons”),elm::list把单个元素elm接在列表list的前面。

  val fruit = "apples" :: ("oranges" :: ("pears" :: Nil))
  val nums = 1 :: (2 :: (3 :: (4 :: Nil)))
  val diag3 = (1 :: (0 :: (0 :: Nil))) ::
              (0 :: (1 :: (0 :: Nil))) ::
              (0 :: (0 :: (1 :: Nil))) :: Nil
  val empty = Nil

由于操作符::是右结合性,所以:

A :: (B :: C)

相当于:

A :: B :: C

所以前一个例子中很多括号都可以省略:

  val nums = 1 :: 2 :: 3 :: 4 :: Nil

列表的基本操作

三个基本操作:headtailisEmpty

  val fruit = "apples" :: "oranges" :: "pears" :: Nil
  val nums = 1 :: 2 :: 3 :: 4 :: Nil
  val diag3 = (1 :: (0 :: (0 :: Nil))) ::
              (0 :: (1 :: (0 :: Nil))) ::
              (0 :: (0 :: (1 :: Nil))) :: Nil
  val empty = Nil
  
empty.isEmpty // true
fruit.isEmpty // flase
fruit.head // "apples"
fruit.tail.head // "organges"
diag3.head // List(1, 0, 0)

headtail只能用在非空列表上,不然抛异常:

  scala> Nil.head
  java.util.NoSuchElementException: head of empty list

一个排序的例子,使用插入排序:对于非空列表x::xs可以先排序xs。然后再把x插入正确的地方:

  def isort(xs: List[Int]): List[Int] =
    if (xs.isEmpty) Nil
    else insert(xs.head, isort(xs.tail))

  def insert(x: Int, xs: List[Int]): List[Int] =
    if (xs.isEmpty || x <= xs.head) x :: xs
    else xs.head :: insert(x, xs.tail)

列表模式

简单的模式匹配,在确定长度的情况下取出列表里的元素:

  scala> val List(a, b, c) = fruit
  a: String = apples
  b: String = oranges
  c: String = pears

不确定具体长度但知道至少有几个,或是只要取前几个:

  scala> val a :: b :: rest = fruit
  a: String = apples
  b: String = oranges
  rest: List[String] = List(pears)

要注意这里的List(...)::并不是之前定义的模式匹配。

实际上List(...)是将来会在抽取器章节介绍的抽取器模式。

“cos”模式x::xs是中缀操作符模式的特例,一般中缀表达式p op q视为p.op(q)。但是如果作为模式,其实是被当作构造器模式的op(p,q)形式。

对应这个构造器模式的类是scala.::,它可以创建非空列表的类。还有一个List类的方法::用来实例化scala.::的对象。在将来的“实现列表”章节中会有进一步的描述。

再次用模式匹配的方式来实现前面已经实现过的插入排序法:

  def isort(xs: List[Int]): List[Int] = xs match {
    case List() => List()
    case x :: xs1 => insert(x, isort(xs1))
  }

  def insert(x: Int, xs: List[Int]): List[Int] = xs match {
    case List() => List(x)
    case y :: ys => if (x <= y) x :: xs
                    else y :: insert(x, ys)
  }

List类的一阶方法

这里介绍的方法是List类的方法,所以是在独立的对象上被调用。

连接列表

连接两个列表的操作符是:::,例如:

  scala> List(1, 2) ::: List(3, 4, 5)
  res0: List[Int] = List(1, 2, 3, 4, 5)

  scala> List() ::: List(1, 2, 3)
  res1: List[Int] = List(1, 2, 3)

  scala> List(1, 2, 3) ::: List(4)
  res2: List[Int] = List(1, 2, 3, 4)

它也是右结合的:

xs ::: ys ::: zs

相当于:

xs ::: (ys ::: zs)

分治原则

手动实现一个连接列表的append方法。先用模式匹配把输入的列表拆分为更加简单的样本:

  def append[T](xs: List[T], ys: List[T]): List[T] =
    xs match {
      case List() => ys
      case x :: xs1 => x :: append(xs1, ys)
    }

以上代码的让ys操持完整而xs被一步步拆分并放到ys前面,所以把注意集中到xs的模式匹配上。

再通过递归调用层层套用剩下的元素,通过添加单个元素的方法::连接列表。

列表长度

  scala> List(1, 2, 3).length
  res3: Int = 3

length方法要遍历整个列表来取得长度,所以判断是否为空一般用isEmpty而不用length

取头和尾

head取头,tail取的是除了第一个元素外剩下列表。这两个方法的运行时间是常量。

last取尾,init取最后一个以外的列表。这两个方法会遍历整个列表。

  scala> val abcde = List('a', 'b', 'c', 'd', 'e')
  abcde: List[Char] = List(a, b, c, d, e)

  scala> abcde.last
  res4: Char = e

  scala> abcde.init
  res5: List[Char] = List(a, b, c, d)

对于空列表会抛异常

  scala> List().init
  java.lang.UnsupportedOperationException: Nil.init
   at scala.List.init(List.scala:544)
   at ...

  scala> List().last
  java.util.NoSuchElementException: Nil.last
   at scala.List.last(List.scala:563)
   at ...

反转列表

reverse是创建了一个新列表:

  scala> abcde.reverse
  res6: List[Char] = List(e, d, c, b, a)

  scala> abcde
  res7: List[Char] = List(a, b, c, d, e)

一些简单的规律:

xs.reverse.reverse equals xs

xs.reverse.init equals xs.tail.reverse
xs.reverse.tail equals xs.init.reverse
xs.reverse.head equals xs.last
xs.reverse.last equals xs.head

通过连接操作:::来实现反转,当然这样的效率低得很:

  def rev[T](xs: List[T]): List[T] = xs match {
    case List() => xs
    case x :: xs1 => rev(xs1) ::: List(x)
  }

前缀与后缀

takedrop取得或舍去列表指定长度个元素,长度超过时不会抛异常而是返回整个列表或空列表。

  scala> abcde take 2
  res8: List[Char] = List(a, b)

  scala> abcde drop 2
  res9: List[Char] = List(c, d, e)

splitAt在指定位置拆分列表。

xs splitAt n

// equals

(xs take n, xs drop n)

例:

  scala> abcde splitAt 2
  res10: (List[Char], List[Char]) = (List(a, b),List(c, d, e))

取得指定元素

通过索引取得指定元素:

  scala> abcde apply 2 // rare in Scala
  res11: Char = c

  scala> abcde(2) // rare in Scala
  res12: Char = c

includes方法取得所有的索引列表:

  scala> abcde.indices
  res13: List[Int] = List(0, 1, 2, 3, 4)

zip

把两个列表组成对偶(二元组),如果长度不一样会丢弃长出来的:

  scala> abcde.indices zip abcde
  res14: List[(Int, Char)] = List((0,a), (1,b), (2,c), (3,d),
  (4,e))

  scala> val zipped = abcde zip List(1, 2, 3)
  zipped: List[(Char, Int)] = List((a,1), (b,2), (c,3))

如果是为了把元素和索引zip在一起,用zipWithIndex方法更有效:

  scala> abcde.zipWithIndex
  res15: List[(Char, Int)] = List((a,0), (b,1), (c,2), (d,3), (e,4))

toString 和 mkString

toString简单字符串化列表

  scala> abcde.toString
  res16: String = List(a, b, c, d, e)

mkString通过三个参数来指定前后包列表的字符和分隔列表元素的字符:

xs mkString (pre, sep, post)

还有两个变体:

xs mkString sep
// equals
xs mkString ("", sep, "")

sx mkString
// equals
xs mkString ""

例子:

  scala> abcde mkString ("[", ",", "]")
  res17: String = [a,b,c,d,e]

  scala> abcde mkString ""
  res18: String = abcde

  scala> abcde.mkString
  res19: String = abcde

  scala> abcde mkString ("List(", ", ", ")")
  res20: String = List(a, b, c, d, e)

还有一个addSting变体让结果添加到StringBuilder中,而不是作为结果返回:

  scala> val buf = new StringBuilder
  buf: StringBuilder =

  scala> abcde addString (buf, "(", ";", ")")
  res21: StringBuilder = (a;b;c;d;e)

列表的转换

List类的toArrayArray类的toList,列表和数组转来转去。

  scala> val arr = abcde.toArray
  arr: Array[Char] = Array(a, b, c, d, e)

  scala> arr.toString
  res22: String = Array(a, b, c, d, e)

  scala> arr.toList
  res23: List[Char] = List(a, b, c, d, e)

copyToArray把列表复制到数组中一会连续的空间内:

  xs copyToArray (arr, start)

start为开始的位置。当然还要保证数组中有足够的空间。例子:

  scala> val arr2 = new Array[Int](10)
  arr2: Array[Int] = Array(0, 0, 0, 0, 0, 0, 0, 0, 0, 0)

  scala> List(1, 2, 3) copyToArray (arr2, 3)

  scala> arr2.toString
  res25: String = Array(0, 0, 0, 1, 2, 3, 0, 0, 0, 0)

elements提供了通过枚举器访问列表元素的方法:

  scala> val it = abcde.elements
  it: Iterator[Char] = non-empty iterator

  scala> it.next
  res26: Char = a

  scala> it.next
  res27: Char = b

例:归并排序

归并排序:如果列表长度为0或是1,就算是已经排序好的,直接返回。长度大于1的列表可以拆成两个长度接近的,每个再递归调用完成排序,再把返回的两个排序好的列表合并。

函数的实现用到了柯里化,接收元素之间的比较大小的函数和要排序的列表:

  def msort[T](less: (T, T) => Boolean)
      (xs: List[T]): List[T] = {

    def merge(xs: List[T], ys: List[T]): List[T] =
      (xs, ys) match {
        case (Nil, _) => ys
        case (_, Nil) => xs
        case (x :: xs1, y :: ys1) =>
          if (less(x, y)) x :: merge(xs1, ys)
          else y :: merge(xs, ys1)
      }

    val n = xs.length / 2
    if (n == 0) xs
    else {
      val (ys, zs) = xs splitAt n
      merge(msort(less)(ys), msort(less)(zs))
    }
  }

使用的方法:

  scala> msort((x: Int, y: Int) => x < y)(List(5, 7, 1, 3))
  res28: List[Int] = List(1, 3, 5, 7)

作为一个柯里化的例子,可以用下划线代表末指定的参数列表:

  scala> val intSort = msort((x: Int, y: Int) => x < y) _
  intSort: (List[Int]) => List[Int] = <function>

如果要改成倒序排序的话,只要换个比较函数:

  scala> val reverseIntSort = msort((x: Int, y: Int) => x > y) _
  reverseIntSort: (List[Int]) => List[Int] = <function>

上面的intSortreverseIntSort都已经绑定了排序的方法,只要传入待排序的列表:

  scala> val mixedInts = List(4, 1, 9, 0, 5, 8, 3, 6, 2, 7)
  mixedInts: List[Int] = List(4, 1, 9, 0, 5, 8, 3, 6, 2, 7)

  scala> intSort(mixedInts)
  res0: List[Int] = List(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)

  scala> reverseIntSort(mixedInts)
  res1: List[Int] = List(9, 8, 7, 6, 5, 4, 3, 2, 1, 0)

List类的高阶函数

这里介绍的方法是List类的方法,所以是在独立的对象上被调用。

Scala中以操作符形式出现的高阶函数更加简洁地处理Java中用循环来处理的问题。

列表间映射

xs map fun把列表中每个元素用方法处理过后生成新列表。xs代表List[T]fun代表T => U的函数。

  scala> List(1, 2, 3) map (_ + 1)
  res29: List[Int] = List(2, 3, 4)

  scala> val words = List("the", "quick", "brown", "fox")
  words: List[java.lang.String] = List(the, quick, brown, fox)
 
  scala> words map (_.length)
  res30: List[Int] = List(3, 5, 5, 3)

  scala> words map (_.toList.reverse.mkString)
  res31: List[String] = List(eht, kciuq, nworb, xof)

flatMapmap类似,但它把所有元素连成一个列表:

  scala> words map (_.toList)
  res32: List[List[Char]] = List(List(t, h, e), List(q, u, i,
      c, k), List(b, r, o, w, n), List(f, o, x))

  scala> words flatMap (_.toList)
  res33: List[Char] = List(t, h, e, q, u, i, c, k, b, r, o, w,
      n, f, o, x)

flatMapmap合作建立出所有1 <= j < i < 5(i, j)对偶:

  scala> List.range(1, 5) flatMap (
       | i => List.range(1, i) map (j => (i, j))
       | )
  res34: List[(Int, Int)] = List((2,1), (3,1), (3,2), (4,1),
      (4,2), (4,3))

上面的代码List.range(1, 5)产生从1到5的整数列表。对于其中的每项i再产生1到i的列表。map产生(i, j)元组列表,这里的j<iflatMpa对每个1到5之间的i产列表,并连接所有列表得到结果。

等同于以下循环结构:

  for (i <- List.range(1, 5); j <- List.range(1, i)) yield (i, j)

foreach没有返回结果(或返回Unit)。如下对sum变量累加,但是没有返回值:

  scala> var sum = 0
  sum: Int = 0

  scala> List(1, 2, 3, 4, 5) foreach (sum += _)

  scala> sum
  res36: Int = 15

过滤

xs filter pxs代表List[T]p代表T => Boolean形式的函数。返回符合的结果列表:

  scala> List(1, 2, 3, 4, 5) filter (_ % 2 == 0)
  res37: List[Int] = List(2, 4)

  scala> words filter (_.length == 3)
  res38: List[java.lang.String] = List(the, fox)

partition方法返回的是所有符合的元素和所有不符合的元素两个列表对。

xs partition p
// equals
( xs filter p , xs filter (!p(_)) )

例:

  scala> List(1, 2, 3, 4, 5) partition (_ % 2 == 0)
  res39: (List[Int], List[Int]) = (List(2, 4),List(1, 3, 5))

find方法只返回第一个符合的元素,一个都不符合返回None

  scala> List(1, 2, 3, 4, 5) find (_ % 2 == 0)
  res40: Option[Int] = Some(2)

  scala> List(1, 2, 3, 4, 5) find (_ <= 0)
  res41: Option[Int] = None

takeWhile不断累积符合的结果直到遇到不符合的;dropWhile不断丢弃不符的元素直到遇到符合的。

  scala> List(1, 2, 3, -4, 5) takeWhile (_ > 0)
  res42: List[Int] = List(1, 2, 3)

  scala> words dropWhile (_ startsWith "t")
  res43: List[java.lang.String] = List(quick, brown, fox)

span方法组合了takeWhiledropWhile返回一对列表,就像是splitAt组合了takedrop一样。

xs span p
// equals
(xs takeWhile p , xs dropWhile p)

split一样,span避免对列表的二次访问:

  scala> List(1, 2, 3, -4, 5) span (_ > 0)
  res44: (List[Int], List[Int]) = (List(1, 2, 3),List(-4, 5))

列表论断

xs forall p全部符合,xs exits p存在符合的元素。

  scala> def hasZeroRow(m: List[List[Int]]) =
       | m exists (row => row forall (_ == 0))
  hasZeroRow: (List[List[Int]])Boolean

  scala> hasZeroRow(diag3)
  res45: Boolean = false

折叠列表

左折叠操作符/:,格式为:(z /: xs) (op)。其中z为开始值,xs为列表,op为二元操作。

(z /: List(a, b, c)) (op)
// equals
op(op(op(z,a), b), c)

用树表示:

      op
     / \
    op c
   / \
  op b
 / \
z a

举例:

  scala> def sum(xs: List[Int]): Int = (0 /: xs) (_ + _)
  sum: (List[Int])Int
  
  scala> sum(List(1, 2, 3)) // equals 0 + 1 + 2 + 3
  res1: Int = 6

  scala> def product(xs: List[Int]): Int = (1 /: xs) (_ * _)
  product: (List[Int])Int
  
  scala> product(List(1, 2, 3)) // equals 1 * 1 * 2 * 3
  res2: Int = 6

用空格连接所有单词:

  scala> ("" /: words) (_ +" "+ _)
  res46: java.lang.String = the quick brown fox

头上多了一个空格,这样去掉它:

  scala> (words.head /: words.tail) (_ +" "+ _)
  res47: java.lang.String = the quick brown fox

相对的还有右倾斜操作树:\

(List(a, b, c) :\ z) (op)
// equals
op(a, op(b, op(c, z)))

对于组合操作来说,左右折叠是等价的,但效率上有差异。下面两个把元素连接在一起的方法:

  def flattenLeft[T](xss: List[List[T]]) =
      (List[T]() /: xss) (_ ::: _)

  def flattenRight[T](xss: List[List[T]]) =
      (xss :\ List[T]()) (_ ::: _)

采用右折叠的flattenLeft需要复制第一个元素列表xss.head一共xss.length-1次,所以效率差一些。

注意这里两个版本的实现都要对作为折叠开始值的空列表做类型标注。这是由Scala类型推断的局限性无法推断出正确的类型。不标注的话会有以下错误:

  scala> def flattenRight[T](xss: List[List[T]]) =
       | (xss :\ List()) (_ ::: _)
  <console>:15: error: type mismatch;
   found : List[T]
   required: List[Nothing]
             (xss :\ List()) (_ ::: _)
                                ^

在以后的“实现列表”章节中讨论类型推断失败的原因。

如果觉得/::\看起来不清楚,可以用List提供的foldLeftfoldRight方法代替。

例子:使用折叠实现列表反转

def reverseLeft[T](xs: List[T]) = (startvalue /: xs) (operation)

为了写出正确的startvalueoperation,从可以出现的最小的列表List()开始推导:

List()
// equals
reverseLeft(List())
// equals
(startvalue /: List()) (operation)
// equals
startvalue

所以startvalue一定是List()。再代入推导operation

List(x)
// equals
reverseLeft(List(x))
// equals
(startvalue /: List(x)) (operation)
// equals
operation(List(), x)
// equals
x :: List()

所以具体实现为:

  def reverseLeft[T](xs: List[T]) =
    (List[T]() /: xs) {(ys, y) => y :: ys}

排序

xs sort beforexs是列表,before是比较元素x是否在y前面的方法。

  scala> List(1, -3, 4, 2, 6) sort (_ < _)
  res48: List[Int] = List(-3, 1, 2, 4, 6)

  scala> words sort (_.length > _.length)
  res49: List[java.lang.String] = List(quick, brown, fox, the)

注意前面还提到过一个msort方法,那个是定义在列表外的。sort是List类的方法。

List对象的方法

下面介绍的方法是伴生对象scala.List的,创建列表的工厂方法和特定类型列表的操作。

通过元素创建列表

apply方法:

List(1, 2, 3)
// is actually
List.apply(1, 2, 3)

按数值范围创建列表

range参数可以是:开始、结束、步长:

  scala> List.range(1, 5)
  res51: List[Int] = List(1, 2, 3, 4)

  scala> List.range(1, 9, 2)
  res52: List[Int] = List(1, 3, 5, 7)

  scala> List.range(9, 1, -3)
  res53: List[Int] = List(9, 6, 3)

创建重复元素的列表

make方法:

  scala> List.make(5, 'a')
  res54: List[Char] = List(a, a, a, a, a)

  scala> List.make(3, "hello")
  res55: List[java.lang.String] = List(hello, hello, hello)

解除Zip列表

unzip把二元组列表分成两个列表:

  scala> val zipped = "abcde".toList zip List(1, 2, 3)
  zipped: List[(Char, Int)] = List((a,1), (b,2), (c,3))

  scala> List.unzip(zipped)
  res56: (List[Char], List[Int]) = (List(a, b, c),
      List(1, 2, 3))

Scala类型系统要求类方法能处理所有类型,而unzip只处理二元组列表。所以unzip不能像zip方法一样放在类里而只能放在伴生对象里。

连接列表

flatten方法只能处理包含子列表的列表,所以不能放在List类里。只能放在伴生对象中。

  scala> val xss =
       | List(List('a', 'b'), List('c'), List('d', 'e'))
  xss: List[List[Char]] = List(List(a, b), List(c), List(d, e))

  scala> List.flatten(xss)
  res57: List[Char] = List(a, b, c, d, e)

concat方法把多个列表作为可变长参数形式接收:

  scala> List.concat(List('a', 'b'), List('c'))
  res58: List[Char] = List(a, b, c)

  scala> List.concat(List(), List('b'), List('c'))
  res59: List[Char] = List(b, c)
  
  scala> List.concat()
  res60: List[Nothing] = List()

映射与测试配对

map2方法接收两个列表,分别作为方法的两个参数:

  scala> List.map2(List(10, 20), List(3, 4, 5)) (_ * _)
  res61: List[Int] = List(30, 80)

exist2也是接收两个列表,分别作为方法的两个参数:

  scala> List.forall2(List("abc", "de"),
       | List(3, 2)) (_.length == _)
  res62: Boolean = true

  scala> List.exists2(List("abc", "de"),
       | List(3, 2)) (_.length != _)
  res63: Boolean = false

了解Scala的类型推断方法

下面是用占位符_推导出的参数类型:

  scala> abcde sort (_ > _)
  res65: List[Char] = List(e, d, c, b, a)

但是msort方法却不能用占位符:

  scala> msort((x: Char, y: Char) => x > y)(abcde)
  res64: List[Char] = List(e, d, c, b, a)
  
  scala> msort(_ > _)(abcde)
  <console>:12: error: missing parameter type for expanded
  function ((x$1, x$2) => x$1.$greater(x$2))
         msort(_ > _)(abcde)
               ^

因为Scala的类型推导器是基于流的。对于`func(args)`这样的方法,先看func是否有已经的类型。如果有的话这个类型就被用来做参数预期类型的推断。

例如对于List[Char]类型的列表abcdabcd的成员都是Char所以abcd.sort(_ > _)的两个参数也只会是Char。所以:

(_ > _)
// trans to
((x: Char, y: Char) => x > y)

而对于msort(_ > _)(abcde)这个类型是柯里化的、多态的方法类型,参数类型是(T, T) => Boolean,返回类型是从List[T]List[T]的函数。无法推断第一个参数的类型。所以类型推断器要参数的类型信息。

想要用占位符的话,只能把参数类型传给msort

  scala> msort[Char](_ > _)(abcde)
  res66: List[Char] = List(e, d, c, b, a)

还有一个方法是交换参数顺序,这样可以用第一个列表的类型来推断比较方法的类型了:

  // same implementation as msort,
  // but with arguments swapped
  def msortSwapped[T](xs: List[T])(less:
      (T, T) => Boolean): List[T] = {
  }
  
  scala> msortSwapped(abcde)(_ > _)
  res67: List[Char] = List(e, d, c, b, a)

需要推断多态方法类型时只会参考第一个参数列表,所以在柯里化方法有两个参数列表时第二个参数不会用来决定方法类型参数。

所以这种方案隐含以下的库方法设计原则:

如果参数包括若干个非函数参数与一个函数参数的组合时,要把函数参数独自放在柯里化参数列表的最后面。这样方法的正确实例类型就可以通过非函数参数推断出来,推断出来的类型还可以转面用来完成函数参数的类型检查。调用函数的时候也可以写出更加简洁的字面量。

再来看更加复杂的折叠操作:

  (xss :\ List[T]()) (_ ::: _)

上面的表达式提供了明确的类型参数的原因是这个右折叠操作的类型取决于两个变量:

  (xs :\ z) (op)

这里把列表xs的类型记为A,如:xs: List[A];而开始值z有可能是类型B。对应的操作op必须以AB的值为参数并返回类型B的结果,即:op: (A, B) => B

从上面的描述可以看出:这里的op方法要知道AB两个类型。A一定与List有关但是B不一定与List有关,所以推不出来。所以下面的表达式是编译不过的:

  (xss :\ List()) (_ ::: _) // this won't compile

上面表达式中z的类型为List[Nothing],据此推断器把B的类型定为Nothing

  (List[T], List[Nothing]) => List[Nothing]

这就意味着输出与输出都是空列表。

就是因为这个问题,所以在柯里化的方法中,方法类型只取决于第一段参数。但是如果不这样做的话,推断器还是没有办法取得op的类型。所以只能程序员明确指定类型。

所以Scala采用的局部的、基于流的类型推断方法还是比较有局限性的;不如ML或是Haskell采用的更加全局化的Hindley-Milner类型推断方式。但是对于面向对象的分支类型处理比Hindley-Mlner更加优雅。由于这些局限性在比较极端的情况下才遇到,所以就在极端情况下明确标类型吧。

另外在遇到多态类型错误时,添加上你认为应该是正确的类型标注也是一种排错方式。

集体类型

概览

scala包中主要特质Iterable,三个子特质:

  • Seq:有序集合。
  • Set:对于==方法不可重复的元素集合。
  • Map:键值映射。

特技Iterable有个抽象方法elements

  def elements: Iterator[A]

注意返回类型是一个迭代器iterator,不是iterate别看错了!

迭代器用来从头到尾遍历一遍集合。如果要再遍历一遍的话,只能用elements方法再生成一个新的迭代器。。

迭代器Iterator继承自AnyRefIterator提供的具体方法都实现了nexthasNext抽象方法实现:

  def hasNext: Boolean
  def next: A

序列

列表

列表不能通过索引直接访问元素,只能遍历;但可以支持在头上快速添加和删除。这点像是链式表。

在头上快速添加和删除元素很好地适合模式匹配。

但是因为只能对列表头快速访问,而尾部不行。所以如果要操作尾部的话可以先建一个反序的列表,再reverse把顺序反过来。

列表缓存

还有一人方式是使用scala.collection.mutable.ListBuffer

+=在尾部添加元素;+:加在头上;完成之后用toList生成List

  scala> import scala.collection.mutable.ListBuffer
  import scala.collection.mutable.ListBuffer

  scala> val buf = new ListBuffer[Int]
  buf: scala.collection.mutable.ListBuffer[Int] = ListBuffer()

  scala> buf += 1

  scala> buf += 2

  scala> buf
  res11: scala.collection.mutable.ListBuffer[Int]
    = ListBuffer(1, 2)

  scala> 3 +: buf
  res12: scala.collection.mutable.Buffer[Int]
    = ListBuffer(3, 1, 2)

  scala> buf.toList
  res13: List[Int] = List(3, 1, 2)

List结合前置添加元素和递归算法增长列表时,如果用的递归算法不是尾递归,就有栈溢出的风险;而ListBuffer可以结合循环替代递归。

数组

数组适合按索引快速访问元素。

按长度产数组:

  scala> val fiveInts = new Array[Int](5)
  fiveInts: Array[Int] = Array(0, 0, 0, 0, 0)

按元素产数组:

  scala> val fiveToOne = Array(5, 4, 3, 2, 1)
  fiveToOne: Array[Int] = Array(5, 4, 3, 2, 1)

通过()指定索引:

  scala> fiveInts(0) = fiveToOne(4)

  scala> fiveInts
  res1: Array[Int] = Array(1, 0, 0, 0, 0)

数组缓存

ArrayBuffer可以在头尾添加元素:

  scala> import scala.collection.mutable.ArrayBuffer
  import scala.collection.mutable.ArrayBuffer

  scala> val buf = new ArrayBuffer[Int]()
  buf: scala.collection.mutable.ArrayBuffer[Int] =
    ArrayBuffer()

  scala> buf += 12
  scala> buf += 15

  scala> buf
  res16: scala.collection.mutable.ArrayBuffer[Int] =
    ArrayBuffer(12, 15)

  scala> buf.length
  res17: Int = 2

  scala> buf(0)
  res18: Int = 12

队列

不可变的队列:

  scala> import scala.collection.immutable.Queue
  import scala.collection.immutable.Queue

  scala> val empty = new Queue[Int]
  empty: scala.collection.immutable.Queue[Int] = Queue()

// add one element
  scala> val has1 = empty.enqueue(1)
  has1: scala.collection.immutable.Queue[Int] = Queue(1)

// use collection to add many elements
  scala> val has123 = has1.enqueue(List(2, 3))
  has123: scala.collection.immutable.Queue[Int] = Queue(1,2,3)

  scala> val (element, has23) = has123.dequeue
  element: Int = 1
  has23: scala.collection.immutable.Queue[Int] = Queue(2,3)

注意上面取后一个出队操作dequeue返回的是一个二元组,包括出来的元素和剩下的队列。

可变的队列也差不多,就是用+=++=添加元素,dequeue方法只返回一个出除的元素。

  scala> import scala.collection.mutable.Queue
  import scala.collection.mutable.Queue

  scala> val queue = new Queue[String]
  queue: scala.collection.mutable.Queue[String] = Queue()

  scala> queue += "a"

  scala> queue ++= List("b", "c")

  scala> queue
  res21: scala.collection.mutable.Queue[String] = Queue(a, b, c)

  scala> queue.dequeue
  res22: String = a

  scala> queue
  res23: scala.collection.mutable.Queue[String] = Queue(b, c)

可变的栈:

  scala> import scala.collection.mutable.Stack
  import scala.collection.mutable.Stack

  scala> val stack = new Stack[Int]
  stack: scala.collection.mutable.Stack[Int] = Stack()

  scala> stack.push(1)

  scala> stack
  res1: scala.collection.mutable.Stack[Int] = Stack(1)

  scala> stack.push(2)

  scala> stack
  res3: scala.collection.mutable.Stack[Int] = Stack(1, 2)

  scala> stack.top
  res8: Int = 2

  scala> stack
  res9: scala.collection.mutable.Stack[Int] = Stack(1, 2)

  scala> stack.pop
  res10: Int = 2

  scala> stack
  res11: scala.collection.mutable.Stack[Int] = Stack(1)

不可变的栈略。



字符串

因为Predef包含了从StringRichString的隐式转换,所以可以把任务字符串当作Seq[Char]

  scala> def hasUpperCase(s: String) = s.exists(_.isUpperCase)
  hasUpperCase: (String)Boolean

  scala> hasUpperCase("Robert Frost")
  res14: Boolean = true

  scala> hasUpperCase("e e cummings")
  res15: Boolean = false

exists方法不在String里,所以隐匿转换为包含exists方法的RichString类。

Set与Map

因为Predef对象通过type关键字指定默认引用了Set与Map的不可变版本:

  object Predef {
    type Set[T] = scala.collection.immutable.Set[T]
    type Map[K, V] = scala.collection.immutable.Map[K, V]
    val Set = scala.collection.immutable.Set
    val Map = scala.collection.immutable.Map
    // ...
  }

所以可变版的要手动声明:

  scala> import scala.collection.mutable
  import scala.collection.mutable

  scala> val mutaSet = mutable.Set(1, 2, 3)
  mutaSet: scala.collection.mutable.Set[Int] = Set(3, 1, 2)

使用Set

Set的关键在于用对象的==检查唯一性。

例子:统计出现的单词

用正则 [ !,.]+ 分隔成单词:

  scala> val text = "See Spot run. Run, Spot. Run!"
  text: java.lang.String = See Spot run. Run, Spot. Run!

  scala> val wordsArray = text.split("[ !,.]+")
  wordsArray: Array[java.lang.String] =
     Array(See, Spot, run, Run, Spot, Run)

建立Set并存入:

  scala> val words = mutable.Set.empty[String]
  words: scala.collection.mutable.Set[String] = Set()

  scala> for (word <- wordsArray)
       | words += word.toLowerCase

  scala> words
  res25: scala.collection.mutable.Set[String] =
    Set(spot, run, see)

常用方法:

val nums = Set(1, 2, 3) (返回)
nums + 5 (返回)
nums - 3 (返回)
nums ++ List(5, 6) (返回)
nums -- List(1, 2) (返回)
nums ** Set(1, 3, 5, 7) 交集(返回Set(3))
nums.size (返回)
nums.contains(3) (返回)
import scala.collection.mutable (返回)
val words = mutable.Set.empty[String] 创建空的可变集
words += "the" (返回)
words -= "the" (返回)
words ++= List("do", "re", "mi") (返回)
words --= List("do", "re") (返回)
words.clear (返回)

Map

使用可变的Map:

  scala> val map = mutable.Map.empty[String, Int]
  map: scala.collection.mutable.Map[String,Int] = Map()

  scala> val map = mutable.Map.empty[String, Int]
  map: scala.collection.mutable.Map[String,Int] = Map()

  scala> map("hello") = 1

  scala> map("there") = 2

  scala> map
  res28: scala.collection.mutable.Map[String,Int] =
    Map(hello -> 1, there -> 2)

  scala> map("hello")
  res29: Int = 1

统计单词出现次数的例子:

  scala> def countWords(text: String) = {
       | val counts = mutable.Map.empty[String, Int]
       | for (rawWord <- text.split("[ ,!.]+")) {
       | val word = rawWord.toLowerCase
       | val oldCount =
       | if (counts.contains(word)) counts(word)
       | else 0
       | counts += (word -> (oldCount + 1))
       | }
       | counts
       | }
  countWords: (String)scala.collection.mutable.Map[String,Int]
  

  scala> countWords("See Spot run! Run, Spot. Run!")
  res30: scala.collection.mutable.Map[String,Int] =
    Map(see -> 1, run -> 3, spot -> 2)

常用方法:

val nums = Map("i"->1, "ii"->2) 返回
nums + ("vi"->6) 返回
nums - "ii" 返回
nums ++ List("iii"->3, "v"->5) 返回
nums -- List("i", "ii") 返回
nums.size 返回
nums.contains("ii") 返回
nums("ii") 返回
nums.keys 返回迭代器
nums.keySet 返回
nums.values 返回
nums.isEmpty 返回
import scala.collection.mutable 返回
val words = mutalbe.Map.empty[String, Int] 返回
words += ("one"->1) 返回
words -= "one" 返回
words ++= List("one"->1, "two"->2, "three"->3)) 返回
words --= List("one", "two") 返回

默认的Set和Map

不可变的类会对数量优化一些工厂方法返回的默认的实现。

不可变的scala.collection.immutable.Set()工厂方法返回:

元素的数量 实现
0 scala.collection.immutable.EmptySet
1 scala.collection.immutable.Set1
2 scala.collection.immutable.Set2
3 scala.collection.immutable.Set3
4 scala.collection.immutable.Set4
>=5 scala.collection.immutable.HashSet

不可变的scala.collection.immutable.Map()工厂方法返回:

元素的数量 实现
0 scala.collection.immutable.EmptyMap
1 scala.collection.immutable.Map1
2 scala.collection.immutable.Map2
3 scala.collection.immutable.Map3
4 scala.collection.immutable.Map4
>=5 scala.collection.immutable.HashMap

有序的集体和映射

TreeSetTreeMap分别实现了SortedSetSortedMap特质。都用红黑树保存元素,顺序由Ordered特质决定。这些类只有不可变的版本:

  scala> import scala.collection.immutable.TreeSet
  import scala.collection.immutable.TreeSet

  scala> val ts = TreeSet(9, 3, 1, 8, 0, 2, 7, 4, 6, 5)
  ts: scala.collection.immutable.SortedSet[Int] =
    Set(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)

  scala> val cs = TreeSet('f', 'u', 'n')
  cs: scala.collection.immutable.SortedSet[Char] = Set(f, n, u)


  scala> import scala.collection.immutable.TreeMap
  import scala.collection.immutable.TreeMap

  scala> var tm = TreeMap(3 -> 'x', 1 -> 'x', 4 -> 'x')
  tm: scala.collection.immutable.SortedMap[Int,Char] =
    Map(1 -> x, 3 -> x, 4 -> x)

  scala> tm += (2 -> 'x')

  scala> tm
  res38: scala.collection.immutable.SortedMap[Int,Char] =
    Map(1 -> x, 2 -> x, 3 -> x, 4 -> x)

同步的Set和Map

SynchronizedMap特质混入到实现中。下面单例对象中的makeMap方法:

  import scala.collection.mutable.{Map,
      SynchronizedMap, HashMap}

  object MapMaker {

    def makeMap: Map[String, String] = {

        new HashMap[String, String] with
            SynchronizedMap[String, String] {

          override def default(key: String) =
            "Why do you want to know?"
        }
    }
  }

上面的方法会返回一个HashMap并且重写了default方法在没有对应的key时有默认的返回。

单线程访问的情况如下:

  scala> val capital = MapMaker.makeMap
  capital: scala.collection.mutable.Map[String,String] = Map()

  scala> capital ++ List("US" -> "Washington",
       | "Paris" -> "France", "Japan" -> "Tokyo")
  res0: scala.collection.mutable.Map[String,String] =
    Map(Paris -> France, US -> Washington, Japan -> Tokyo)

  scala> capital("Japan")
  res1: String = Tokyo

  scala> capital("New Zealand")
  res2: String = Why do you want to know?

  scala> capital += ("New Zealand" -> "Wellington")

  scala> capital("New Zealand")
  res3: String = Wellington

类似的同步的Set:

  val synchroSet =
    new mutable.HashSet[Int] with
        mutable.SynchronizedSet[Int]

可变与不可变类型的比较

为了方便在可变与不可变类型之间地转换,Scala提供了一些语法糖。

如,不可变类型不支持+=操作:

  scala> val people = Set("Nancy", "Jane")
  people: scala.collection.immutable.Set[java.lang.String] =
    Set(Nancy, Jane)

  scala> people += "Bob"
  <console>:6: error: reassignment to val
         people += "Bob"
                ^

但是如果把变量从val改成var,Scala还是可以返回一个添加后的新对象来模拟:

  scala> var people = Set("Nancy", "Jane")
  people: scala.collection.immutable.Set[java.lang.String] =
    Set(Nancy, Jane)
 
  scala> people += "Bob"

  scala> people
  res42: scala.collection.immutable.Set[java.lang.String] =
    Set(Nancy, Jane, Bob)

类似的还有其他的操作:

  scala> people -= "Jane"

  scala> people ++= List("Tom", "Harry")

  scala> people
  res45: scala.collection.immutable.Set[java.lang.String] =
    Set(Nancy, Bob, Tom, Harry)

这样的语法糖方便在可变与不可变类型之间转换:

  var capital = Map("US" -> "Washington", "France" -> "Paris")
  capital += ("Japan" -> "Tokyo")
  println(capital("France"))

  import scala.collection.mutable.Map // only change needed!
  var capital = Map("US" -> "Washington", "France" -> "Paris")
  capital += ("Japan" -> "Tokyo")
  println(capital("France"))

这样的语法糖还可以用在其他类型上。如浮点:

  scala> var roughlyPi = 3.0
  roughlyPi: Double = 3.0

  scala> roughlyPi += 0.1

  scala> roughlyPi += 0.04

  scala> roughlyPi
  res48: Double = 3.14

基本上+=-=*=这类以=结尾的操作符都可以。

初始化集合

最典型的是用伴生对象的工厂方法:

  scala> List(1, 2, 3)
  res0: List[Int] = List(1, 2, 3)

  scala> Set('a', 'b', 'c')
  res1: scala.collection.immutable.Set[Char] = Set(a, b, c)

  scala> import scala.collection.mutable
  import scala.collection.mutable

  scala> mutable.Map("hi" -> 2, "there" -> 5)
  res2: scala.collection.mutable.Map[java.lang.String,Int] =
    Map(hi -> 2, there -> 5)

  scala> Array(1.0, 2.0, 3.0)
  res3: Array[Double] = Array(1.0, 2.0, 3.0)

会根据工厂方法推断类型:

  scala> import scala.collection.mutable
  import scala.collection.mutable

  scala> val stuff = mutable.Set(42)
  stuff: scala.collection.mutable.Set[Int] = Set(42)

  scala> stuff += "abracadabra"
  <console>:7: error: type mismatch;
   found : java.lang.String("abracadabra")
   required: Int
         stuff += "abracadabra"
                  ^

但是可以手动声明类型:

  scala> val stuff = mutable.Set[Any](42)
  stuff: scala.collection.mutable.Set[Any] = Set(42)

还有一种情况,不能直接把List传递给Set的工厂方法:

  scala> val colors = List("blue", "yellow", "red", "green")
  colors: List[java.lang.String] =
    List(blue, yellow, red, green)


  scala> import scala.collection.immutable.TreeSet
  import scala.collection.immutable.TreeSet

  scala> val treeSet = TreeSet(colors)
  <console>:6: error: no implicit argument matching
    parameter type (List[java.lang.String]) =>
      Ordered[List[java.lang.String]] was found.
         val treeSet = TreeSet(colors)
                       ^

可行的方案是建立空的TreeSet[String]对象并用TreeSet++操作把元素加进去:

  scala> val treeSet = TreeSet[String]() ++ colors
  treeSet: scala.collection.immutable.SortedSet[String] =
     Set(blue, green, red, yellow)

数组与列表之间转换

  scala> treeSet.toList
  res54: List[String] = List(blue, green, red, yellow)

  scala> treeSet.toArray
  res55: Array[String] = Array(blue, green, red, yellow)

Set与Map的可变与不可变互转

在转为不可变类型时,一般是建一个空的不可变集,再一个一个加上去:

  scala> import scala.collection.mutable
  import scala.collection.mutable

  scala> treeSet
  res5: scala.collection.immutable.SortedSet[String] =
    Set(blue, green, red, yellow)

  scala> val mutaSet = mutable.Set.empty ++ treeSet
  mutaSet: scala.collection.mutable.Set[String] =
    Set(yellow, blue, red, green)

  scala> val immutaSet = Set.empty ++ mutaSet
  immutaSet: scala.collection.immutable.Set[String] =
    Set(yellow, blue, red, green)

  scala> val muta = mutable.Map("i" -> 1, "ii" -> 2)
  muta: scala.collection.mutable.Map[java.lang.String,Int] =
     Map(ii -> 2, i -> 1)

  scala> val immu = Map.empty ++ muta
  immu: scala.collection.immutable.Map[java.lang.String,Int] =
     Map(ii -> 2, i -> 1)

元组

元组可以存放不同的类型:

  (1, "hello", Console)

元组经常被用来返回多个函数结果,如下面的函数要同时返回单词和索引:

  def longestWord(words: Array[String]) = {
    var word = words(0)
    var idx = 0
    for (i <- 1 until words.length)
      if (words(i).length > word.length) {
        word = words(i)
        idx = i
      }
    (word, idx)
  }
  
  scala> val longest =
       | longestWord("The quick brown fox".split(" "))
  longest: (String, Int) = (quick,1)

然后可以访问各个元素:

  scala> longest._1
  res56: String = quick

  scala> longest._2
  res57: Int = 1

还可以赋值给自己的变量(其实就是模式匹配):

  scala> val (word, idx) = longest
  word: String = quick
  idx: Int = 1

  scala> word
  res58: String = quick

注意括号不能去掉,不然就是给两个变量赋值了两份:

  scala> val word, idx = longest
  word: (String, Int) = (quick,1)
  idx: (String, Int) = (quick,1)

有状态的对象

类似于JavaBean的getter和setter方法,Scala对象的非私有var x有自动生成的访问方法x和设值方法x_=

对于类中的字段:

var hour = 12

会有额外的getter方法hour和setter方法hour_=。方法的访问性与字段一致。

拿这个例子来说:

  class Time {
    var hour = 12
    var minute = 0
  }

和下面的代码是一样的:

  class Time {

    private[this] var h = 12
    private[this] var m = 0

    def hour: Int = h
    def hour_=(x: Int) { h = x }

    def minute: Int = m
    def minute_=(x: Int) { m = x }
  }

所以可以直接定义getter和setter。

下面的代码在setter前进行检查:

  class Time {

    private[this] var h = 12
    private[this] var m = 12

    def hour: Int = h
    def hour_= (x: Int) {
      require(0 <= x && x < 24)
      h = x
    }

    def minute = m
    def minute_= (x: Int) {
      require(0 <= x && x < 60)
      m = x
    }
  }

再看一个温度的例子:

  class Thermometer {

    var celsius: Float = _

    def fahrenheit = celsius * 9 / 5 + 32
    def fahrenheit_= (f: Float) {
      celsius = (f - 32) * 5 / 9
    }
    override def toString = fahrenheit +"F/"+ celsius +"C"
  }

注意变量celsius的值为_,表示初始化值。对于数值代表0,对于布尔类型代表false,引用类型则代表null

Scala中的初始化器=_,如果写成:

var celsius

这样就成了抽象变量(以后到了“抽象成员”这一章介绍),而不是一个没有初始化的变量。这个和Java的习惯很不一样。

使用的例子:

  scala> val t = new Thermometer
  t: Thermometer = 32.0F/0.0C

  scala> t.celsius = 100

  scala> t
  res3: Thermometer = 212.0F/100.0C

  scala> t.fahrenheit = -40

  scala> t
  res4: Thermometer = -40.0F/-40.0C

案例:离散事件模拟

来个SICP(Structure and Interpretation of Computer Programs,计算机程序的构造与解释)里的例子。

为数字电路定制语言

为了实现这三种基本的门,我们建立一个Wire类代表线路。可以这样构造线路:

val a = new Wire
val b = new Wire
val c = new Wire

或简洁地写成:

val a, b, c = new Wire

三个基本的门电路由以下三个过程模拟:

  def inverter(input: Wire, output: Wire)
  def andGate(a1: Wire, a2: Wire, output: Wire)
  def orGate(o1: Wire, o2: Wire, output: Wire)

注意这里的过程都没有返回值。按照函数式的思想,应该是返回构造好的门对象。但是在这里我们选择了没有返回值,而是通过副作用来模拟门电路。

在这副作用让一步步渐进地构造复杂的电路更加容易,如inverter(a,b)ab之间放置反转电路。

还有这里的方法名没有用动词而是用了名词,这是为了方便说明制造的是哪个门电路。这反映了DSL说明的本质:应该描述电路,而不是如何制造它。

下面是一个半加法器(half-adder)。它根据两个输入ab产生累加和s

累加的定义为:s= (a+b)%2及进位c,其中的c = (a+b)/2

半加法器电路图:

用我们的代码描述:

  def halfAdder(a: Wire, b: Wire, s: Wire, c: Wire) {
    val d, e = new Wire
    orGate(a, b, d)
    andGate(a, b, c)
    inverter(c, e)
    andGate(d, e, s)
  }

接下来是一个全加法器,定义为根据参数ab还有进位cin得到两个输出。一个是和sum = (a+b+cin)%2,另一个是进位输出count = (a+b+cin)/2

代码为:

  def fullAdder(a: Wire, b: Wire, cin: Wire,
      sum: Wire, cout: Wire) {

    val s, c1, c2 = new Wire
    halfAdder(a, cin, s, c1)
    halfAdder(b, s, sum, c2)
    orGate(c1, c2, cout)
  }

这是内部DSL很好的例子:通过宿主语言将特定的语言定义为库面不是完全实现这种语言。

Simulation API

在我们的例子中,把参数列表和返回都为空的过程() => Unit作为基本的动作。给这样类型的过程起个别名叫Action

type Action = () => Unit

私有变量保存时间,但提供对时间的公开访问:

  private var curtime: Int = 0

  def currentTime: Int = curtime

在特定时间执行的的操作定义为工作项目(work item):

  case class WorkItem(time: Int, action: Action)

注意这里用的是样本类,所以用工厂方法创建实例就可以自动获得访问构造器参数timeaction的方法。

还有一个类来保存末执行工作条目的排程表(agenda),它是按时间排序的:

  private var agenda: List[WorkItem] = List()

提供在一定 时延后加入新的工作条目的方法,加入操作也要排序:

  def afterDelay(delay: Int)(block: => Unit) {
    val item = WorkItem(currentTime + delay, () => block)
    agenda = insert(agenda, item)
  }

  private def insert(ag: List[WorkItem],
      item: WorkItem): List[WorkItem] = {

    if (ag.isEmpty || item.time < ag.head.time) item :: ag
    else ag.head :: insert(ag.tail, item)
  }

核心是run方法:

  def run() {
    afterDelay(0) {
      println("*** simulation started, time = "+
          currentTime +" ***")
    }
    while (!agenda.isEmpty) next()
  }
  
  private def next() {
    (agenda: @unchecked) match {
      case item :: rest =>
        agenda = rest
        curtime = item.time
        item.action()
    }
  }

注意这里为了方便去掉了空列表的情况。为了防止编译器警告我们在模式匹配里故意漏掉了列表为空的情况,在这里使用了(agenda: @unchecked) match而不是agenda match

完整的代码在包org.stairwaybook.simulation里:

  abstract class Simulation {

    type Action = () => Unit

    case class WorkItem(time: Int, action: Action)

    private var curtime = 0
    def currentTime: Int = curtime

    private var agenda: List[WorkItem] = List()

    private def insert(ag: List[WorkItem],
        item: WorkItem): List[WorkItem] = {

      if (ag.isEmpty || item.time < ag.head.time) item :: ag
      else ag.head :: insert(ag.tail, item)
    }

    def afterDelay(delay: Int)(block: => Unit) {
      val item = WorkItem(currentTime + delay, () => block)
      agenda = insert(agenda, item)
    }

    private def next() {
      (agenda: @unchecked) match {
        case item :: rest =>
          agenda = rest
          curtime = item.time
          item.action()
      }
    }

    def run() {
      afterDelay(0) {
        println("*** simulation started, time = "+
            currentTime +" ***")
      }
      while (!agenda.isEmpty) next()
    }
  }

电路模拟

这里创建了BasicCircuitSiomulation来模拟电路。

为了模拟电路和延迟声明了三个方法:InverterDelayAndGateDelayOrGateDelay。由于不同模拟电路的技术参数不同,所以这三个方法是抽象方法。

Wire类

需要支持的三种基本动作:

getSignal: Boolean:返回当前线路上的信号。

setSignal(sig: Boolean):设置线路信号。

addAction(p: Action):添加动作到线路上。基本思想是所有附加在某线路上的动作过程在每次信号改变时被执行。通过连接组件可以为线路添加该组件的功能。加上的动作会在被加到线路时以及每次线路信号改变时被执行。

实现代码sigVal代表当前信号,actions是附加的动作过程。需要注意的是setSignal方法,当信号改变时,新的信号首先被保存在变量sigVal中,然后执行所有线路附加动作:

  class Wire {
    private var sigVal = false
    private var actions: List[Action] = List()

    def getSignal = sigVal

    def setSignal(s: Boolean) =
      if (s != sigVal) {
        sigVal = s
        actions foreach (_ ())
      }

    def addAction(a: Action) = {
      actions = a :: actions
      a()
    }
  }

注意上面的缩写格式:actions forearch(_())代表对每个元素执行_()。在“函数和装饰”这一章的“占位符”部分说明过,函数_()f => f()的缩写,代表空参数函数。

反转操作

inverter方法会在安装之后以及每次线路信号变化时被调用。它通过setSignal把输出设为输入的反值。

另外,由于还要模拟电路的响应时间,所以输入值改变以后,还要等InverterDelay单位的模拟时间后,才发生改变:

  def inverter(input: Wire, output: Wire) = {
    def invertAction() {
      val inputSig = input.getSignal
      afterDelay(InverterDelay) {
        output setSignal !inputSig
      }
    }
    input addAction invertAction
  }

注意这里的afterDelay方法是把这个操作加到队列的最后面。

与门和或门操作

大致思想和上面类似:

  def andGate(a1: Wire, a2: Wire, output: Wire) = {
    def andAction() = {
      val a1Sig = a1.getSignal
      val a2Sig = a2.getSignal
      afterDelay(AndGateDelay) {
        output setSignal (a1Sig & a2Sig)
      }
    }
    a1 addAction andAction
    a2 addAction andAction
  }

  def orGate(o1: Wire, o2: Wire, output: Wire) {
    def orAction() {
      val o1Sig = o1.getSignal
      val o2Sig = o2.getSignal
      afterDelay(OrGateDelay) {
        output setSignal (o1Sig | o2Sig)
      }
    }
    o1 addAction orAction
    o2 addAction orAction
  }

模拟输出

通过探针(probe)观察线路上信号的改变。

还是在信号改变时被调用,显示输出线路的名称、模拟时间、信号值:

  def probe(name: String, wire: Wire) {
    def probeAction() {
      println(name +" "+ currentTime +
          " new-value = "+ wire.getSignal)
    }
    wire addAction probeAction
  }

运行模拟器

BasicCircuitSimulation继承了CircuitSimulation

  package org.stairwaybook.simulation

  abstract class CircuitSimulation
    extends BasicCircuitSimulation {

    def halfAdder(a: Wire, b: Wire, s: Wire, c: Wire) {
      val d, e = new Wire
      orGate(a, b, d)
      andGate(a, b, c)
      inverter(c, e)
      andGate(d, e, s)
    }

    def fullAdder(a: Wire, b: Wire, cin: Wire,
        sum: Wire, cout: Wire) {

      val s, c1, c2 = new Wire
      halfAdder(a, cin, s, c1)
      halfAdder(b, s, sum, c2)
      orGate(c1, c2, cout)
    }
  }

剩下的电路延迟时间和定义被模拟的电路都留在Scala交互Shell中实现:

  scala> import org.stairwaybook.simulation._
  import org.stairwaybook.simulation._

定义延迟时间:

  scala> object MySimulation extends CircuitSimulation {
       | def InverterDelay = 1
       | def AndGateDelay = 3
       | def OrGateDelay = 5
       | }
  defined module MySimulation

定义一下简化以后对MySimulation的引用:

  scala> import MySimulation._
  import MySimulation._

定义线路的部分。先定义四根线路,再把探针放在其中的两根上。探针会立即输出结果:

  scala> val input1, input2, sum, carry = new Wire
  input1: MySimulation.Wire =
      simulator.BasicCircuitSimulation$Wire@111089b
  input2: MySimulation.Wire =
      simulator.BasicCircuitSimulation$Wire@14c352e
  sum: MySimulation.Wire =
      simulator.BasicCircuitSimulation$Wire@37a04c
  carry: MySimulation.Wire =
      simulator.BasicCircuitSimulation$Wire@1fd10fa

  scala> probe("sum", sum)
  sum 0 new-value = false

  scala> probe("carry", carry)
  carry 0 new-value = false

加上半加法器:

  scala> halfAdder(input1, input2, sum, carry)

逐次把两根输入线信号设为true,并执行模拟过程:

  scala> input1 setSignal true

  scala> run()
  *** simulation started, time = 0 ***
  sum 8 new-value = true

  scala> input2 setSignal true

  scala> run()
  *** simulation started, time = 8 ***
  carry 11 new-value = true
  sum 15 new-value = false

全部代码如下:

package org.stairwaybook.simulation

abstract class BasicCircuitSimulation extends Simulation {

  def InverterDelay: Int
  def AndGateDelay: Int
  def OrGateDelay: Int

  class Wire {

    private var sigVal = false
    private var actions: List[Action] = List()

    def getSignal = sigVal

    def setSignal(s: Boolean) =
      if (s != sigVal) {
        sigVal = s
        actions foreach (_ ())
      }

    def addAction(a: Action) = {
      actions = a :: actions
      a()
    }
  }

  def inverter(input: Wire, output: Wire) = {
    def invertAction() {
      val inputSig = input.getSignal
      afterDelay(InverterDelay) {
        output setSignal !inputSig
      }
    }
    input addAction invertAction
  }

  // continued in Listing 18.10...
  // ...continued from Listing 18.9
  def andGate(a1: Wire, a2: Wire, output: Wire) = {
    def andAction() = {
      val a1Sig = a1.getSignal
      val a2Sig = a2.getSignal
      afterDelay(AndGateDelay) {
        output setSignal (a1Sig & a2Sig)
      }
    }
    a1 addAction andAction
    a2 addAction andAction
  }

  def orGate(o1: Wire, o2: Wire, output: Wire) {
    def orAction() {
      val o1Sig = o1.getSignal
      val o2Sig = o2.getSignal
      afterDelay(OrGateDelay) {
        output setSignal (o1Sig | o2Sig)
      }
    }
    o1 addAction orAction
    o2 addAction orAction
  }

  def probe(name: String, wire: Wire) {
    def probeAction() {
      println(name +" "+ currentTime +
          " new-value = "+ wire.getSignal)
    }
    wire addAction probeAction
  }
}

abstract class Simulation {

  type Action = () => Unit

  case class WorkItem(time: Int, action: Action)

  private var curtime = 0
  def currentTime: Int = curtime

  private var agenda: List[WorkItem] = List()

  private def insert(ag: List[WorkItem],
      item: WorkItem): List[WorkItem] = {

    if (ag.isEmpty || item.time < ag.head.time) item :: ag
    else ag.head :: insert(ag.tail, item)
  }

  def afterDelay(delay: Int)(block: => Unit) {
    val item = WorkItem(currentTime + delay, () => block)
    agenda = insert(agenda, item)
  }

  private def next() {
    (agenda: @unchecked) match {
      case item :: rest =>
        agenda = rest
        curtime = item.time
        item.action()
    }
  }

  def run() {
    afterDelay(0) {
      println("*** simulation started, time = "+
          currentTime +" ***")
    }
    while (!agenda.isEmpty) next()
  }
}


abstract class CircuitSimulation
  extends BasicCircuitSimulation {

  def halfAdder(a: Wire, b: Wire, s: Wire, c: Wire) {
    val d, e = new Wire
    orGate(a, b, d)
    andGate(a, b, c)
    inverter(c, e)
    andGate(d, e, s)
  }

  def fullAdder(a: Wire, b: Wire, cin: Wire,
      sum: Wire, cout: Wire) {

    val s, c1, c2 = new Wire
    halfAdder(a, cin, s, c1)
    halfAdder(b, s, sum, c2)
    orGate(c1, c2, cout)
  }
}

 object MySimulation extends CircuitSimulation {
           def InverterDelay = 1
           def AndGateDelay = 3
           def OrGateDelay = 5

  def main(args: Array[String]) {
    val input1, input2, sum, carry = new Wire

    probe("sum", sum)
    probe("carry", carry)
    halfAdder(input1, input2, sum, carry)

    input1 setSignal true
    run()

    input2 setSignal true
    run()
  }
}
Category: scala | Tags: scala
3
31
2013
58

Scala学习笔记-Part.01

配置

字符编码问题

在默认字符编码为UTF-8的Linux下没问题。

Mac OS X系统的默认字符编码早就改成了UTF-8但它bundle的Java默认字符编码却一直是MacRoman。在启动REPL时传入参数-Dfile.encoding=UTF-8

用vim、emacs或者你习惯的文本编辑器打开scala命令,比如:

$ vim `which scala`

找到如下行:

[ -n "$JAVA_OPTS" ] || JAVA_OPTS="-Xmx256M -Xms32M"

把-D参数加到JAVA_OPTS里即可。

Vim插件

https://github.com/scala/scala-dist][相关工具上下载,复制tool-support/src/vim.vim目录下。

使用

Scala Shell

使用进入REPL环境的方式:

--(morgan-laptop:pts/8)-(13-03-15 9:04:57)-(~/workspace/study/scala)
\-(morgan:%) >>> scala
Welcome to Scala version 2.9.2 (OpenJDK 64-Bit Server VM, Java 1.7.0_15).
Type in expressions to have them evaluated.
Type :help for more information.

scala> 

输入回车自动换行。

发现输错了,再按几个回车就退出了。

退出Scala Shell:quit:q

在REPL环境中只能一行一行读取,所以如果要换行的话,不能让一行在语法上看起来已经结束:

scala> if(x > 0) { 1
     | } else if(x == 0) 0 else -1
res1: Int = 1

另一个方法是在REPL中输入:paste粘贴代码,按下Control + D

脚本

脚本文件,可以接收一个参数并输出欢迎信息:

/* 可以接收一个参数 */
println("Hello, " + args(0) + "!")

调用脚本:scala命令、文件名、参数

--(morgan-laptop:pts/8)-(13-03-14 23:28:27)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> scala helloarg.scala Jade
Hello, Jade!

可以通过循环处理多个参数的:

args.foreach( arg => println(arg) )

调用:

--(morgan-laptop:pts/8)-(13-03-14 23:49:39)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> scala pa.scala Scala is even more fun      
Scala
is
even
more
fun

OS可执行脚本

Unix下可执行脚本:

#!/bin/sh
	exec scala "$0" "$@"
!#

println("hello," + arg(0) + "!")

执行:

$ chmod +x helloarg

$ ./helloarg globe

Windows下可执行脚本:

::#!
@echo off
call scala %0 %*
goto :eof
::!#

println("hello," + arg(0) + "!")

执行:

> helloarg.bat globe

Scala程序

先看一个工具类,它根据字符串来计算出检验和:

import scala.collection.mutable.Map

class ChecksumAccumulator {
	private var sum = 0
	def add(b: Byte) { sum += b }
	def checksum(): Int =  ~(sum &0xFF) + 1
}

object ChecksumAccumulator {
	private val cache = Map[String, Int]()

	def calculate(s: String): Int =
		if( cache.contains(s) ) {
			cache(s)
		} else {
			val acc = new ChecksumAccumulator
			for (c <- s)
				acc.add(c.toByte)
			val cs = acc.checksum()
			cache += (s -> cs)
			cs
		}
}

然后是主程序。独立运行的程序一定要有main方法(仅有一个参数Array[String]而且结果类型为Unit)的单例对象。

import ChecksumAccumulator.calculate

object Summer {

	def main(args: Array[String]) {
		for (arg <- args)
			println(arg + " -> " + calculate(arg))
	}

}

编译Scala程序:

--(morgan-laptop:pts/8)-(13-03-15 0:28:39)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> scalac ChecksumAccumulator.scala Summer.scala

有一个fast Scala compiler的后台进程可以在第一次被调用后一直跑在后台,节省一下每次编译的速度:

--(morgan-laptop:pts/8)-(13-03-15 0:29:11)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> fsc ChecksumAccumulator.scala Summer.scala   

可以关掉这个后台进程:

--(morgan-laptop:pts/8)-(13-03-15 0:29:11)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> fsc -shutdown

编译完后可以看到生成的.class文件:

--(morgan-laptop:pts/8)-(13-03-15 0:44:31)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> ls *.class
ChecksumAccumulator$$anonfun$calculate$1.class  ChecksumAccumulator$.class    Summer.class
ChecksumAccumulator.class                       Summer$$anonfun$main$1.class  Summer$.class

运行编译出来的类文件:

--(morgan-laptop:pts/8)-(13-03-15 0:30:58)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> scala Summer of love
of -> -213
love -> -182

还有一个加入的Application特质的方式实现应用程序,但是有局限:不能访问命令行参数、只能在单线程下用。所以不推荐用它。

形式类似于:

object MyApp extends Application {
	println("Hello")
}	

Ant任务

相关的Ant任务有scalacfscscaladoc,这里只介绍scalac

scala.home=/opt/morganstudio/language/scala
compile.version=1.7
<?xml version="1.0" encoding="UTF-8"?>
<project name="scala-example" default="init" basedir=".">
	<description>scala example</description>
	<property file="build.properties" />

	<property name="sources.dir" value="sources" />
	<property name="build.dir" value="build" />

	<target name="init">
		<property name="scala-library.jar" 
			value="${scala.home}/lib/scala-library.jar" />
		<path id="build.classpath">
			<pathelement location="${scala-library.jar}"   />
			<pathelement location="${build.dir}"   />
		</path>
		<taskdef resource="scala/tools/ant/antlib.xml">
			<classpath>
				<pathelement location="${scala.home}/lib/scala-compiler.jar"   />
				<pathelement location="${scala-library.jar}"   />
			</classpath>
		</taskdef>
	</target>

	<target name="build" depends="init">
		<mkdir dir="${build.dir}"   />
		<scalac srcdir="${sources.dir}"
			destdir="${build.dir}"
			classpathref="build.classpath">
			<include name="**/*.scala"   />
		</scalac>
	</target>

	<target name="run" depends="build">
		<java classname="Summer"
			classpathref="build.classpath">
		</java>
	</target>
</project>

以后可能会用到的jar包还有scala-actors.jarscala-dbc.jar.

脚本与程序的区别

脚本必须以表达式结束,而程序以定义结尾。尝试以scala解释器运行程序会报错。

入门

变量定义

val不可变变量;var可变变量。格式:

// 基本格式
val msg: java.lang.String = "Hello"

// java.lang默认已经导入了
val msg: String = "Hello"

// 自动推导类型
val msg = "Hello"

语句定义

Scala语句以分号结束,而且分号可以省略。默认一行结束了就是一行语句结束了,除非以下三种情况,会认为语句还没有结束:

  • 行尾是一个不能放在行尾的字符。
  • 下一行的开头是不能放在行头的字符。
  • ()[]里,这里面不能放多条语句。

这是两个:

val s = "hello"; println(s)
if (x < 2)
	println("too small")
else
	println("ok")

这是两个:

x
+ y

一个:

(x
+ y)

一个:

x +
y +
z

函数定义

取最大的函数,函数体的最后一行作为结果返回:

def max(x: Int, y: Int): Int = {
	if (x > y) x else y
}

函数类型也能自动推导出来,可以省略。在递归函数的情况下,一定要明确地说明返回类型。如果函数体只有一行,那花括号也可以省略:

scala> def max(x: Int, y: Int) =  if (x > y) x else y
max: (x: Int, y: Int)Int

类型Unit对应Java中的void。即没有参数又没有返回结果的函数定义:

scala> def greet() = println("Hello")
greet: ()Unit

对于没有等号的方法来说返回类型一定是Unit。Scala可以把任何类型转为Unit,以下方法最后的String类结果会转为Unit并丢弃:

scala> def f(): Unit = "This String is lost!"
f: ()Unit

有花括号但没有等号的方法默认为Unit,有了等号但没有类型会由编译器自动推导:

scala> def f() {"This String is lost!"}
f: ()Unit

scala> def f() = {"This String get returned!"}
f: ()java.lang.String

scala> f
res2: java.lang.String = This String get returned!

函数字面量(Function Literal)

函数字面量用=>来分隔参数表与函数体:

(x:Int, y:Int) => x + y

通过函数字面量来迭代处理参数的例子:

args.foreach( (arg: String) => println(arg) )

//这里的String类型可以自动推导出来:
args.foreach( arg => println(arg) )

//在这种字面量只有一行而且只有一个参数情况下,可以省掉参数列表
args.foreach( println )

for循环

scala> for (i <- 0 to 10) print(i)
012345678910

方法与操作符

下面的语句产一个从0到5的集合:

scala> 0 to 5
res9: scala.collection.immutable.Range.Inclusive = Range(0, 1, 2, 3, 4, 5)

其实这个to是一个方法的调用。Scala中对于方法调用时,如果方法只有一个参数的话可以省略括号,原本的样子是:

(0).to(5)

scala中没有操作符的重载,因为操作符也是方法的名字:

1 + 2

相当于:

(1).+(2)

标识符

Scala在构成标识符方面有四种非常灵活的规则:

字母数字标识符(alphanumeric identifier)

字母数字标识符起始于一个字母或下划线,之后可以跟字母,数字,或下划线。‘\(’字符也被当作是字母,但是被保留作为Scala编译器产生的标识符之用。用户程序里的标识符不应该包含‘\)’字符,尽管能够编译通过;但是这样做有可能导致与Scala编译器产生的标识符发生名称冲撞。

Scala遵循Java的驼峰式标识符习俗。尽管下划线在标识符内是合法的,但在Scala程序里并不常用,部分原因是为了保持与Java一致,同样也由于下划线在Scala代码里有许多其它非标识符用法。因此,最好避免使用像to_string__init__,或name这样的标识符。

标识符结尾使用下划线的一个结果就是,如果你尝试写一个这样的定义:val name_: Int = ,你会收到一个编译器错误。编译器会认为你正常是定义一个叫做name_:的变量。要让它编译通过,你将需要在冒号之前插入一个额外的空格,如:val name_ : Int = 1

字段,方法参数,本地变量,还有函数的驼峰式名称,应该以小写字母开始,如:lengthflatMap,还有s。

类和特质的驼峰式名称应该以大写字母开始,如:BigIntList,还有UnbalancedTreeMap

Scala与Java的习惯不一致的地方在于常量名。Scala里,constant这个词并不等同于val。尽管val在被初始化之后的确保持不变,但它还是个变量。比方说,方法参数是val,但是每次方法被调用的时候这些val都可以代表不同的值。而常量更持久。比方说,scala.Math.Pi被定义为很接近实数π的双精度值,表示圆周和它的直径的比值。这个值不太可能改变,因此Pi显然是个常量。你还可以用常数去给一些你代码里作为幻数(magic number)要用到的值一个名字:文本值不具备解释能力,如果出现在多个地方将会变得极度糟糕。Java里,习惯上常量名全都是大写的,用下划线分隔单词,如MAX_VALUEPI。Scala里,习惯只是第一个字母必须大写。因此,Java风格的常量名,如X_OFFSET,在Scala里也可以用,但是Scala的惯例是常数也用驼峰式风格,如XOffset

操作符标识符(operator identifier)

由一个或多个操作符字符组成。操作符字符是一些如+:?~#的可打印的ASCII字符。以下是一些操作符标识符的例子:

+ ++ ::: <?> :->

Scala编译器将内部“粉碎”操作符标识符以转换成合法的内嵌\(的Java标识符。例如,标识符:->将被内部表达为\)colon\(minus\)greater。若你想从Java代码访问这个标识符,就应使用这个内部表达。

Scala里的操作符标识符可以变得任意长,因此在Java和Scala间有一些小差别。Java里,输入x<-y将会被拆分成四个词汇符号,所以写成x < - y也没什么不同。Scala里,<-将被作为一个标识符拆分,而得到x <- y。如果你想要得到第一种解释,你要在<-字符间加一个空格。这大概不会是实际应用中的问题,因为没什么人会在Java里写x<-y的时候不注意加空格或括号的。

混合标识符(mixed identifier)

混合标识符由字母数字组成,后面跟着下划线和一个操作符标识符。例如,unary_+被用做定义一元的+操作符的方法名。或者,myvar_=被用做定义赋值操作符的方法名。多说一句,混合标识符格式myvar_=是由Scala编译器产生的用来支持属性:property的;第十八章进一步说明。

文本标识符(literal identifier)

文本标识符是用反引号包括的任意字串。如:

`x` `<clinit>` `yield` 

它的思路是你可以把任何运行时认可的字串放在反引号之间当作标识符。结果总是Scala标识符。即使包含在反引号间的名称是Scala保留字,这个规则也是有效的。在Java的Thread类中访问静态的yield方法是其典型的用例。你不能写Thread.yield()因为yield是Scala的保留字。然而,你仍可以在反引号里引用方法的名称,例如:

Thread.`yield`()

类型参数化数组

长度为3的数组,存放的元素类型为String

val gs: Array[String] = new Array[String](3)

scala> val gs = new Array[String](3)
gs: Array[String] = Array(null, null, null)

scala> gs(0) = "aa"
scala> gs(1) = "bb"
scala> gs(2) = "cc"

scala> gs.foreach(print)
aabbcc

scala> val ns = Array("11","22","33")
ns: Array[java.lang.String] = Array(11, 22, 33)

scala> ns.foreach(print)
112233

apply与update方法

对一个对象的后面加上括号的操作其实是调用了这个对象的apply方法。所以数组的元素索引操作其实是apply方法调用:

gs(0)
//相当于:
gs.apply(0)

val ns = Array("11","22","33")
//相当于:
val ns = Array.apply("11","22","33")

对带有括号并包括一到多个参数的变量赋值时,编译器使用对象的update方法对括号里的参数(索引值)和等号右边的对象执行调用:

gs(0) = "aa"
//相当于:
gs.update(0, "aa")

列表

java.util.List不同,scala.List是不可变的。不可变的对象更加符合函数式风格。

scala> val ll = List(1,2,3)
ll: List[Int] = List(1, 2, 3)

::把一个元素加到列表的头上; 用:::连起两个列表:

scala> 0 :: ll
res12: List[Int] = List(0, 1, 2, 3)

scala> val ll2 = List(4,5,6)
ll2: List[Int] = List(4, 5, 6)

scala> ll ::: ll2
res11: List[Int] = List(1, 2, 3, 4, 5, 6)

一个元素也没有的空列表用Nil表示,作为一个空的列表,它可以把其他的元素给串起来:

scala> val nl = 1 :: 2 :: 3 :: Nil
nl: List[Int] = List(1, 2, 3)

List只能把元素加在头上,如果要加在后面的话,一个方法是在加到头上以后再调用reverse方法;还有一个方案是使用ListBuffer,它有append方法。

方法关联性

所有以:结尾的方法其实是后一个操作数调用前一个操作数,所以:

0 :: ll
// 其实是
ll.::(0)

ll ::: ll2
// 其实是
ll2.:::(ll)

回到前面的串列表操作:

val nl = 1 :: 2 :: 3 :: Nil

如果没有最后的Nil,就变成了3.::(2)。因为数字没有::方法,这样就会报错。

List常用方法

List() 或 Nil                          // 空List
List("Cool", "tools", "rule")          // 创建带有三个值"Cool","tools"和"rule"的新List[String]

val thrill = "Will"::"fill"::"until"::Nil  // 创建带有三个值"Will","fill"和"until"的新List[String]

List("a", "b") ::: List("c", "d")      // 叠加两个列表(返回带"a","b","c"和"d"的新List[String])
thrill(2)                              // 返回在thrill列表上索引为2(基于0)的元素(返回"until")
thrill.count(s => s.length == 4)       // 计算长度为4的String元素个数(返回2)
thrill.drop(2)                         // 返回去掉前2个元素的thrill列表(返回List("until"))
thrill.dropRight(2)                    // 返回去掉后2个元素的thrill列表(返回List("Will"))
thrill.exists(s => s == "until")       // 判断是否有值为"until"的字串元素在thrill里(返回true)
thrill.filter(s => s.length == 4)      // 依次返回所有长度为4的元素组成的列表(返回List("Will", "fill"))
thrill.forall(s => s.endsWith("1"))    // 辨别是否thrill列表里所有元素都以"l"结尾(返回true)
thrill.foreach(s => print(s))          // 对thrill列表每个字串执行print语句("Willfilluntil")
thrill.foreach(print)                  // 与前相同,不过更简洁(同上)
thrill.head                            // 返回thrill列表的第一个元素(返回"Will")
thrill.init                            // 返回thrill列表除最后一个以外其他元素组成的列表(返回List("Will", "fill"))
thrill.isEmpty                         // 说明thrill列表是否为空(返回false)
thrill.last                            // 返回thrill列表的最后一个元素(返回"until")
thrill.length                          // 返回thrill列表的元素数量(返回3)
thrill.map(s => s + "y")               // 返回由thrill列表里每一个String元素都加了"y"构成的列表(返回List("Willy", "filly", "untily"))
thrill.mkString(", ")                  // 用列表的元素创建字串(返回"will, fill, until")
thrill.remove(s => s.length == 4)      // 返回去除了thrill列表中长度为4的元素后依次排列的元素列表(返回List("until"))
thrill.reverse                         // 返回含有thrill列表的逆序元素的列表(返回List("until", "fill", "Will"))

thrill.sort((s, t) => s.charAt(0).toLowerCase < t.charAt(0).toLowerCase)
// 返回包括thrill列表所有元素,并且第一个字符小写按照字母顺序排列的列表(返回List("fill", "until", "Will"))

thrill.tail                            // 返回除掉第一个元素的thrill列表(返回List("fill", "until"))

元组(Tuple)

元组像列表,但可以放不同类型的元素。这样类似于Java Bean,但写起来更加简单。元组的类型按字段个数来识别,有2个字段的就是Tuple2、3个就是Tuple3,Scala最多支持到Tuple22

scala> val pair = (99, "Luft")
pair: (Int, java.lang.String) = (99,Luft)

访问字段通过_序号来实现。不能像数组一样用()的原因是:如果要用apply方法,那定义方法的时候就要声明返回类型,而同一个元组中元素的类型是不同的,所以写不出这个apply方法。

scala> print(pair._1)
99
scala> print(pair._2)
Luft

集(Set)和映射(Map)

对于Map和Set,Scala都分别提供了可变和不变的版本(放一不同的包里)。可变版本的操作会在本地修改,不可变的版本会返回一个新的对象。一般默认会使用不可变版本。

Set继承关系:

Set继承关系

scala> var jetSet = Set("Boeing", "Airbus")
jetSet: scala.collection.immutable.Set[java.lang.String] = Set(Boeing, Airbus)

scala> jetSet += "Lear"

scala> println(jetSet.contains("Cessna"))
false

scala> println(jetSet)
Set(Boeing, Airbus, Lear)

有些情况下想要指定使用可变版本的:

scala> import scala.collection.mutable.Set
import scala.collection.mutable.Set

scala> val movieSet = Set("Hitch", "Poltergeist")
movieSet: scala.collection.mutable.Set[java.lang.String] = Set(Poltergeist, Hitch)

scala> movieSet += "Shrek"
res3: movieSet.type = Set(Shrek, Poltergeist, Hitch)

scala> println(movieSet)
Set(Shrek, Poltergeist, Hitch)

指定要使用HashSet:

scala> import scala.collection.immutable.HashSet
import scala.collection.immutable.HashSet

scala> val hashSet = HashSet("Tomatoes", "Chilies")
hashSet: scala.collection.immutable.HashSet[java.lang.String] = Set(Chilies, Tomatoes)

scala> println(hashSet + "Coriander")
Set(Chilies, Tomatoes, Coriander)

Map继承关系:

Map继承关系

默认的Map用不可变的类型:

scala> val romanNumberal = Map( 1 -> "I", 2 -> "II", 3 -> "III",
     | 4 -> "IV", 5 -> "V")
romanNumberal: scala.collection.immutable.Map[Int,java.lang.String] = Map(5 -> V, 1 -> I, 2 -> II, 3 -> III, 4 -> IV)

scala> println(romanNumberal(4))
IV

使用一个可变的Map

scala> import scala.collection.mutable.Map
import scala.collection.mutable.Map

scala> val treasureMap = Map[Int, String]()
treasureMap: scala.collection.mutable.Map[Int,String] = Map()

scala> treasureMap += (1 -> "Go to inland.")
res6: treasureMap.type = Map(1 -> Go to inland.)

scala> treasureMap += (2 -> "Find big X on ground.")
res7: treasureMap.type = Map(1 -> Go to inland., 2 -> Find big X on ground.)

scala> treasureMap += (3 -> "Dig.")
res8: treasureMap.type = Map(3 -> Dig., 1 -> Go to inland., 2 -> Find big X on ground.)

scala> println(treasureMap(2))
Find big X on ground.

函数式风格

函数式风格极力避免使用变量(就是用到变量也尽量用val这种不可变的变量)与副作用。

典型的指令式风格

先来看一个指令式的for循环:

scala> val args = Array("11","22","33")
args: Array[java.lang.String] = Array(11, 22, 33)

scala> def printArgs(args: Array[String]): Unit = {
     | var i = 0
     | while (i < args.length) {
     | println(args(i))
     | i += 1
     | }
     | }
printArgs: (args: Array[String])Unit

去除变量的使用

通过去掉val的使用变得更加函数式风格:

scala> def printArgs(args: Array[String]): Unit = {
     | for (arg <- args) println(arg)
     | }
printArgs: (args: Array[String])Unit

当然更优雅的风格是这样的:

scala> def printArgs(args: Array[String]): Unit = {
     | args.foreach(println)
     | }
printArgs: (args: Array[String])Unit

去除副作用

光去掉了变量的使用还不是函数式的。因为这个例子中还有副作用:打印到输出流。

所以我们在这里把字符串的格式化与打印输出分成两个功能来做:

scala> def formatArgs(args: Array[String]) = args.mkString("\n")
formatArgs: (args: Array[String])String

scala> println(formatArgs(args))
11
22
33

这样才真正算是函数式风格。鼓励程序员尽量设计出没有副作用,没有变量的代码。

读取文本文件

一个读取文本文件的方法,统计每个行里的字符数:

import scala.io.Source

if (args.length > 0) {
	for (line <- Source.fromFile(args(0)).getLines)
		println(line.length + " " + line)
} else {
	Console.err.println("Please enter filename")
}

执行一下:

--(morgan-laptop:pts/5)-(13-03-16 17:49:53)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> scala readFile.scala readFile.scala
22 import scala.io.Source
0 
22 if (args.length > 0) {
48      for (line <- Source.fromFile(args(0)).getLines)
35              println(line.length + " " + line)
8 } else {
45      Console.err.println("Please enter filename")
1 }

执行的结束不错,但是没有排版……强化一下,先遍历一次得到最长的统计参数。

import scala.io.Source

def widthOfLength(s: String) = s.length.toString.length

if (args.length > 0) {
	val lines = Source.fromFile(args(0)).getLines.toList

	/* 找到最长的一行,不用for循环,
	   显得更加函数式一些 */
	val longestLine = lines.reduceLeft(
		(a, b) => if (a.length > b.length) a else b
	)
	val maxWidth = widthOfLength(longestLine)

	for (line <- lines) {
		val numSpaces = maxWidth - widthOfLength(line)
		val padding = " " * numSpaces
		println(padding + line.length + " | " + line)
	}
} else {
	Console.err.println("Please enter filename")
}

输出格式为:

--(morgan-laptop:pts/8)-(13-03-17 15:14:04)-(~/workspace/study/scala/tmp)
\-(morgan:%) >>> scala readFile.scala readFile.scala
22 | import scala.io.Source
 0 | 
55 | def widthOfLength(s: String) = s.length.toString.length
 0 | 
22 | if (args.length > 0) {
53 |    val lines = Source.fromFile(args(0)).getLines.toList
 0 | 
36 |    val longestLine = lines.reduceLeft(
45 |            (a, b) => if (a.length > b.length) a else b
 2 |    )
42 |    val maxWidth = widthOfLength(longestLine)
 0 | 
22 |    for (line <- lines) {
48 |            val numSpaces = maxWidth - widthOfLength(line)
31 |            val padding = " " * numSpaces
47 |            println(padding + line.length + " | " + line)
 2 |    }
 8 | } else {
45 |    Console.err.println("Please enter filename")
 1 | }

基本类型

基本类型包括java.lang包下的Stringscala包下的ByteShortIntLongFloatDoubleCharBoolean。还有在scala.runtime包下对应的包装器类Rich...,如:RichInt

字符串

除了和Java一样的字符串字面量表示方式以外,Scala还提供了原始字符串(raw string)方便照原文解读:

println("""Welcome to Ultamix 3000.
           Type "HELP" for help.""")

输出的内容包括所有的转义字符和空格:

Welcome to Ultamix 3000.
           Type "HELP" for help.

有些情况下希望在源代码里也能排版提好看一点,所以字符串里提供stripMargin方法可以通过管道符|来取得想要的部分:

println("""|Welcome to Ultamix 3000.
           |Type "HELP" for help.""".stripMargin)
Welcome to Ultamix 3000.
Type "HELP" for help.

符号

格式为'symb,这里的symb可以是任何字母或数字。这种字面量被直接映射为类scala.Symbol的实例,解释器调用工厂方法Symbol("symb")产生。

符号变量什么事情都做不了,只能显示自己的名字,而且符号变量是被限定(interned)的,如果同一个字面量出现两次,其实指向的是同一个Symble实例:

scala> val s = 'aSymbol
s: Symbol = 'aSymbol

scala> s.name
res3: String = aSymbol

那符号能用来干嘛?比如说,下面的函数更新记录,field是字段名、value是值:

scala> def updateRecordByName(field: Symbol, value: Any){ }
updateRecordByName: (field: Symbol, value: Any)Unit

scala> updateRecordByName('pcOK, "OK Computer")

操作符与方法

操作符也是普通方法的另一种写法,操作符的重载也就是方法的重载。

中缀操作符(infix)

scala> val s = "Hello, world!"
s: java.lang.String = Hello, world!

scala> s indexOf 'o'
res6: Int = 4

scala> s indexOf ('o', 5)
res7: Int = 8

前缀操作符

前缀操作符以unary_开头,能有四种+-!~

scala> - 2.0
res8: Double = -2.0

scala> (2.0).unary_-
res9: Double = -2.0

其他的符号就算定义了也不能作为前置操作符解释,如果定义了p.unary_*,在调用*p会被Scala解释为*.p

后缀操作符

后缀操作符其实就是没有参数的函数。一般习惯上没有副作用的话就加上括号,如:println();如果没有副作用就不加括号,如:String.toLowerCase

scala> "Hello".toLowerCase
res10: java.lang.String = hello

scala> "Hello" toLowerCase
res11: java.lang.String = hello

对象相等性

操作符==!=不仅比较基本类型,也可以比较对象,甚至是不同类的对象也可以比较,也可以和null比不会有异常抛出:

scala>  1 == 2
res12: Boolean = false

scala> 1 != 2
res13: Boolean = true

scala> List(1, 2, 3) == List(1, 2, 3)
res14: Boolean = true

scala> List(1, 2, 3) == List(4, 5, 6)
res15: Boolean = false

scala> 1 == 1.0
res16: Boolean = true

scala> List(1, 2, 3) == "hello"
res17: Boolean = false

scala> List(1, 2, 3) == null
res18: Boolean = false

scala> null == List(1, 2, 3)
res19: Boolean = false

scala> ("he" + "llo") == "hello"
res20: Boolean = true

简单定义类与创建对象:

scala> class ChecksumAccumulator { }
defined class ChecksumAccumulator

scala> new ChecksumAccumulator
res0: ChecksumAccumulator = ChecksumAccumulator@91f1520

scala> class ChecksumAccumulator {
     | var sum = 0
     | }
defined class ChecksumAccumulator

scala> val acc = new ChecksumAccumulator
acc: ChecksumAccumulator = ChecksumAccumulator@501fdcfb

scala> val csa = new ChecksumAccumulator
csa: ChecksumAccumulator = ChecksumAccumulator@58f285cd

默认访问控制为public。

成员方法:

class ChecksumAccumulator {
	private var sum = 0
	
	def add(b: Byte): Unit = {
		sum += b
	}

	def checksum(): Int =  {
		return ~(sum & 0xFF) + 1
	}
}

Scala中参数都是val,不可变。

	def add(b: Byte): Unit = {
		// b = 1   // error, because b is val
		sum += b
	}

只有一行的方法体可以去掉花括号并放在函数头一行,方法会自动返回最后一行语句,不用加return:

class ChecksumAccumulator {
	private var sum = 0
	def add(b: Byte): Unit = sum += b
	def checksum(): Int =  ~(sum & 0xFF) + 1
}

没有返回的方法可以省略类型Unit与等号:

	def add(b: Byte): Unit = sum += b
	// 简化
	def add(b: Byte) { sum += b }

创建新类型

分数(rational number)表示分子(numerator)和分母(denominator)的比率,其中分母不能为零。能让小数部分得到了完全表达,没有舍入或估算。要模型化分数的行为,包括允许它们执行加,减,乘还有除运算。

要加两个分数,首先要获得公分母,然后才能把两个分子相加。

要乘两个分数,可以简单的两个分子相乘,然后两个分母相乘。

除法是把右操作数分子分母调换,然后做乘法。

一个分数加到另外一个分数上,产生的结果是一个新的分数。而原来的数不会被“改变”。

主构造器:primary constructor

如果类没有主体,就不需要指定一对空的大括号(当然你如果想的话也可以)。

class Rational(n: Int, d: Int)

在类名Rational之后括号里的n和d,被称为类参数(class parameter)。nd并不是类中的字段,而是主构造器的两个参数。Scala编译器会收集这两个类参数并创造一个带同样的两个参数的主构造器(primary constructor)。Java类具有可以带参数的构造器,而Scala类可以直接带参数。

Scala编译器将把你放在类内部的任何不是字段的部分或者方法定义的代码,编译进主构造器。例如:

scala> class Rational(n: Int, d: Int) { println("Created "+n+"/"+d) }

scala> new Rational(1, 2)
Created 1/2 res0: Rational = Rational@a0b0f5

你可以像这样打印输出一条消息,因为打印语句也在主构造器中。

先决条件(precondition)

先决条件是对传递给方法或构造器的值的限制,是调用者必须满足的需求。使用Predef包中的require方法。如果传入的值为真,require将正常返回。反之,require将通过抛出IllegalArgumentException来阻止对象被构造。

class Rational(n: Int, d: Int) { 
	require(d != 0) 
}

字段

要注意的是:

scala> class Rational(n: Int, d: Int)

在前面主构造器部分已经提别提到:nd并不是类中的字段,而是主构造器的两个参数。所以下面人代码是无法访问到nd的:

def showRational(r: Rational): Rational = println("Rational: "+n+"/"+d) 

所以又增加了两个字段,分别是numer和denom,并用类参数n和d初始化它们:

class Rational(n: Int, d: Int) {
	require(d != 0) 

	val numer: Int = n 
	val denom: Int = d 
}

在对象外面访问分子和分母:

scala> val r = new Rational(1, 2) 
r: Rational = 1 / 2 

scala> r.numer 
res7: Int = 1 

scala> r.denom 
res8: Int = 2

方法

添加加法运算,得到另外一个分数后返回一个新对象为二者的和:

class Rational(n: Int, d: Int) {
	require(d != 0) 

	val numer: Int = n 
	val denom: Int = d 

	def add(that: Rational): Rational = new Rational( 
		numer * that.denom + that.numer * denom, 
		denom * that.denom 
	)
}

加法操作:

scala> val oneHalf = new Rational(1, 2) 
oneHalf: Rational = 1/2 

scala> val twoThirds = new Rational(2, 3) 
twoThirds: Rational = 2/3 

scala> oneHalf add twoThirds 
res0: Rational = 7/6

自指向

关键字this指向当前执行方法被调用的对象实例,或者如果使用在构造器里的话,就是正被构建的对象实例。

例如,我们考虑添加一个方法,lessThan,来测试给定的分数是否小于传入的参数:

def lessThan(that: Rational) = 
	this.numer * that.denom < that.numer * this.denom 

这里,this.numer指向lessThan被调用的那个对象的分子。你也可以去掉this前缀而只是写numer;着两种写法是相同的。

举一个不能缺少this的例子,考虑在Rational类里添加max方法返回指定分数和参数中的较大者:

def max(that: Rational) = 
	if (this.lessThan(that)) that else this

这里,第一个this是冗余的,你写成(lessThan(that))也是一样的。但第二个this表示了当测试为假的时候的方法的结果;如果你省略它,就什么都返回不了了。

从构造器

有些时候一个类里需要多个构造器。Scala里主构造器之外的构造器被称为从构造器(auxiliary constructor)。Scala的从构造器开始于def this(...)

Java里,构造器的第一个动作必须要么调用同类里的另一个构造器,要么直接调用超类的构造器。Scala的类里面,只有主构造器可以调用超类的构造器。Scala里更严格的限制实际上是权衡了更高的简洁度和与Java构造器相比的简易性所付出的代价之后作出的设计。

比方说,分母为1的分数只写分子的话就更为简洁。如,对于5/1来说,可以只是写成5。因此,如果不是写成Rational(5, 1),客户程序员简单地写成Rational(5)或许会更好看一些。这就需要给Rational添加一个只带一个参数分子的从构造器并预先设定分母为1。

class Rational(n: Int, d: Int) { 
	require(d != 0) 

	val numer: Int = n 
	val denom: Int = d 

	def this(n: Int) = this(n, 1)

Rational的从构造器主体几乎完全是调用主构造器,直接传递了它的唯一的参数n作为分子和1作为分母。

私有字段和方法

分数的分子分母有时可以约掉,添加一个最大公约数的私有方法:

class Rational(n: Int, d: Int) { 
	require(d != 0) 

	private val g = gcd(n.abs, d.abs) 

	val numer = n / g 
	val denom = d / g 

	private def gcd(a: Int, b: Int): Int = 
		if (b == 0) a else gcd(b, a % b) 
}

定义操作符

用通常的数学的符号替换add方法,同样实现一个*方法以实现乘法:

def +(that: Rational): Rational = new Rational( 
	numer * that.denom + that.numer * denom, 
	denom * that.denom 

def *(that: Rational): Rational = 
	new Rational(numer * that.numer, denom * that.denom)

使用

scala> val x = new Rational(1, 2)
x: Rational = 1/2

scala> val y = new Rational(2, 3) 
y: Rational = 2/3 

scala> x.+(y) 
res33: Rational = 7/6

scala> x + y 
res32: Rational = 7/6

而且实现的加法和乘法都带有优先级:

scala> x + x * y 
res34: Rational = 5/6 

scala> (x + x) * y 
res35: Rational = 2/3 

scala> x + (x * y) 
res36: Rational = 5/6

方法覆盖(override)

override修饰符标示了之前的方法定义被重载。Rational类里增加toString方法的方式重载:override缺省的实现,如:

class Rational(n: Int, d: Int) { override def toString = n +"/"+ d } 

方法定义前的override修饰符标示了之前的方法定义被重载;第10章会更进一步说明。现在分数显示得很漂亮了,所以我们去掉了前一个版本的Rational类里面的println除错语句。你可以在解释器里测试Rational的新行为:

scala> val x = new Rational(1, 3) 
x: Rational = 1/3 

scala> val y = new Rational(5, 7) 
y: Rational = 5/7

方法重载(overload)

方法的参数表不同产生重载。

给每个数学方法都有两个版本了:一个带分数做参数,另一个带整数。

def +(that: Rational): Rational = new Rational( 
    numer * that.denom + that.numer * denom, 
    denom * that.denom 
  ) 

def +(i: Int): Rational = new Rational(numer + i * denom, denom) 

def -(that: Rational): Rational = new Rational( 
    numer * that.denom - that.numer * denom, 
    denom * that.denom 
  ) 

def -(i: Int): Rational = new Rational(numer - i* denom, denom) 

def *(that: Rational): Rational = new Rational(
    numer * that.numer, 
    denom * that.denom
  ) 
 
def *(i: Int): Rational = new Rational(numer * i, denom) 

def /(that: Rational): Rational = new Rational(
   numer * that.denom, 
    denom * that.numer
  ) 
 
def /(i: Int): Rational = new Rational(numer, denom * i)

隐式转换

虽然现在可以写r * 2了,但是不能用2 * r这样的写法:

scala> val x = new Rational(2, 3)

scala> 2 * r
error: overloaded method value * with alternatives:
  (x: Double)Double <and>
  (x: Float)Float <and>
  (x: Long)Long <and>
  (x: Int)Int <and>
  (x: Char)Int <and>
  (x: Short)Int <and>
  (x: Byte)Int
 cannot be applied to (this.Rational)

出错的原因是因为Int类上没有重载我们自己建的Rational类乘法。

解决的方案是告诉Scala如何把Int类转换为Rational类,再加上修饰符implicit通知Scala编译器可以自动调用:

scala> implicit def intToRational(x: Int) = new Rational(x)

scala> 2 * r
res16: Rational = 4/3

隐式转换只能在定义的作用范围内起作用,如果隐式方法被定义在Rational类中,就不在解释器的作用范围内,所以要把它定义在解释器内。

完整的Rational代码

class Rational(n: Int, d: Int) { 
	require(d != 0) 

	private val g = gcd(n.abs, d.abs)
	val numer = n / g 
	val denom = d / g 

	def this(n: Int) = this(n, 1)

	def +(that: Rational): Rational = new Rational( 
    numer * that.denom + that.numer * denom, 
    denom * that.denom 
  ) 

  def +(i: Int): Rational = new Rational(numer + i * denom, denom) 

  def -(that: Rational): Rational = new Rational( 
      numer * that.denom - that.numer * denom, 
      denom * that.denom 
    ) 

  def -(i: Int): Rational = new Rational(numer - i* denom, denom) 

  def *(that: Rational): Rational = new Rational(
      numer * that.numer, 
      denom * that.denom
    ) 

  def *(i: Int): Rational = new Rational(numer * i, denom) 

  def /(that: Rational): Rational = new Rational(
     numer * that.denom, 
     denom * that.numer
    ) 

  def /(i: Int): Rational = new Rational(numer, denom * i)

	def lessThan(that: Rational) = 
		this.numer * that.denom < that.numer * this.denom 

	def max(that: Rational) = 
		if (this.lessThan(that)) that else this

	override def toString = n +"/"+ d

	private def gcd(a: Int, b: Int): Int = 
		if (b == 0) a else gcd(b, a % b) 
}

val x = new Rational(2, 3)
print("    x = ");  println(x)
print("x * x = ");  println(x * x)
print("x * 2 = ");  println(x * 2)

implicit def intToRational(x: Int) = new Rational(x)
print("2 * x = ");  println(2 * x)

单例对象

Scala中没有像Java那样的静态成员而是用单例对象(Singleton Object)来代替。

在定义格式基本上和类一样,除了了object关键字代替class:

object ObjNam{
	// ...
}

伴生对象

如果一个单例对象的名字和类一样,并且必须在同一个文件里。那它就是这个类的伴生对象(Companion Object),类是它的伴生类(Companion Class)。它们可以相互访问私有成员。

import scala.collection.mutable.Map

class ChecksumAccumulator {
	private var sum = 0
	def add(b: Byte) { sum += b }
	def checksum(): Int =  ~(sum & 0xFF) + 1
}

object ChecksumAccumulator {
	private val cache = Map[String, Int]()

	def caculate(s: String): Int = {
		if (cache.contains(s))
			cache(s)
		else {
			val acc = new ChecksumAccumulator
			for (c <- s)
				acc.add(c.toByte)
			val cs = acc.checksum()
			cache += (s -> cs)
			cs
		}
	}
}

直接通过调用方法:

val res1 = ChecksumAccumulator.caculate("Every value is an object")
println(res1)

val res2 = ChecksumAccumulator.caculate("So simple!")
println(res2)

内建控制结构

if表达式

if表达式返回执行分支的结果:

var filename = "default.txt"
if (!args.isEmpty)
	filename = args(0)

更加函数式地写法,去掉变量:

val filename = 
	if (!args.isEmpty) args(0)
	else "default.txt"

在没有副作用的情况下,用变量的目的就是为了存个值。而表达式就是算值的,所以直接拿表达式来用得了:

println(if (!args.isEmpty) args(0) else "default.txt")

while循环

用do-while算最大公约数:

def gcdLoop(x: Long, y: Long): Long = {
	var a = x; var b = y
	while (a != 0) {
		val temp = a; a = b % a; b = temp
	}
	b
}

用while-do读取文件:

var line = ""
do {
	line = readline()
		println("Read: " + line)
} while (line != "")

while循环与Unit

注意这里没有叫它“表达式”。原因是它不会有返回值(类型为Unit,写作“()”),所以不是表达式。

scala> def greet() { println("HI") }
greet: ()Unit

scala> greet() == ()
<console>:9: warning: comparing values of types Unit and Unit using `==' will always yield true
              greet() == ()
                      ^
HI
res1: Boolean = true

scala> () == ""
<console>:8: warning: comparing values of types Unit and java.lang.String using `==' will always yield false
              () == ""
                 ^
res2: Boolean = false

scala> () != ""
<console>:8: warning: comparing values of types Unit and java.lang.String using `!=' will always yield true
              () != ""
                 ^
res3: Boolean = true

注意上面的警告信息:Unit与Unit进行相等运算永远为true;与String相等运算永远false;与String不相等运算永远为true。

现在说到重点了:Scala中var赋值操作也是Unit,而不是和Java一样返回变量值。所以下面这种Java中一直用到的写法在Scala中是会出问题的:

var line = ""
while ((line = readline()) != "") {  // always true !!!
		println("Read: " + line)
} 

while循环与函数式风格

由于while没有返回值,所以常常被函数式语言舍弃。例如对于同样一个求最大公约数的函数,对比一下指令式与函数式的区别。

指令式,用循环:

def gcdLoop(x: Long, y: Long): Long = {
	var a = x; var b = y
	while (a != 0) {
		val temp = a; a = b % a; b = temp
	}
	b
}

函数式,用递归:

def gcd(x: Long, y: Long): Long = if (y == 0) x else gcd(x, x % y)

while循环是没有返回值的,那就一定要用副作用:不是更新var就是写I/O之类的,不然浪费电么?

for表达式

Scale中的for表达式是遍历集合类的强大工具,包括了过滤与构造新集合的功能。

生成器(generator)

生成器语法item <- collection,对应每步遍历时集合中对应的项:

// Array[file]
val fileList = (new java.io.File(".")).listFiles

for (file <- fileList) println(file)

指定次数:

scala> for (i <- 1 to 4) print(i + " ")
1 2 3 4

scala> for (i <- 1 until 4) print(i + " ")
1 2 3 

不推荐的风格,遍历时还要考虑下标是从0还是从1开始,会不会越界:

for (i <-0 to filesList.length -1)
	println(fileList(i))

for是表达式

for表达式之所以被称为“表达式”是因为它能产生令人感兴趣的值,一个其类型取决于for表达式<-子句的集合。

过滤器(filter)

只要处理以.scala结尾的文件:

for (file <- fileList)
	if ( file.getName.endsWith(".scala") )
		println(file)

Scala可以做得更好,给for循环加上选择器:

for (file <- fileList if file.getName.endsWith(".scala"))
		println(file)

而且不止可以加一个,当然多个语句之间要用分号分隔:

for (
	file <- fileList 
	if file.isFile;
	if file.getName.endsWith(".scala")
) println(file)

for循环嵌套

通过多个<-得到嵌套:

def fileLines(file: java.io.File) = 
	scala.io.Source.fromFile(file).getlines.toList

def grep(pattern: String) =
	for {
		file <- filesList
		if file.getName.endWith(".scala")
		line <- fileLines(file)
		if line.trim.matches(pattern)
	} println(file + ": " + line.trim)

grep(".*gcd.*")

就像上面的代码那样,花括号可以代替小括号。但是替换的目的是什么呢?

在前面提到过的Scala的断句原则时提到,在小括号里被认为是一条语句,所以多个for嵌套时要加上分号分隔;花括号里可以放多条,根据换行可以正确断句。

(流间)变量绑定(mid-stream)

请注意前面的代码段中重复出现的表达式line.trim。这不是个可忽略的计算,因此你或许想每次只算一遍。通过用等号=把结果绑定到新变量可以做到这点。绑定的变量被当作val引入和使用,不过不用带关键字val。

def grep(pattern: String) =
	for {
		file <- filesList if file.getName.endWith(".scala")
		line <- fileLines(file)
		trimmed = line.trim if line.trim.matches(pattern)
	} println(file + ": " + trimmed)

制造新集合

到现在为止所有的例子都只是对枚举值进行操作然后就放过,除此之外,你还可以创建一个值去记住每一次的迭代。只要在for表达式之前加上关键字yield

for表达式在每次执行的时候都会制造一个值,当for表达式完成的时候,结果将是一个包含了所有产生的值的集合。结果集合的类型基于枚举子句处理的集合类型。

对于for-yield表达式的语法是这样的:

for {子句} yield {循环体}

比如,下面的函数鉴别出.scala文件并保存在数组里:

def scalaFiles = 
	for { 
		file <- filesHere if file.getName.endsWith(".scala") 
	} yield file

for表达式在每次执行的时候都会制造一个值,本例中是file。本例中结果为Array[File],因为filesHere是数组并且产生的表达式类型是File。

再来一个取每行长度的例子:

val forLineLengths =
	for {
		file <- filesList if file.getName.endWith(".scala")
		line <- fileLines(file)
		trimmed = line.trim if line.trim.matches(".*for.*")
	} yield trimmed.length

try表达式与异常

抛出异常

throw new IllegalArgumentException

throw也是有结果类型的表达式,而且还可以转换成任何类型。所以可以写在赋值语句里。没有异常就是表达式的值,有异常了得到Nothing

val half = 
	if ( n % 2 == 0 ) n/2
	else throw new RuntimeException("n must be even")

捕获异常

捕获异常的语法选择catch子句的形式。这样设计的原因是为了与Scala很重要的部分:模式匹配(pattern matching)保持一致。模式匹配是一种很强大的特征,将在稍后概述并在另外的章节中详述。

import java.io.FileReader 
import java.io.FileNotFoundException 
import java.io.IOException 

try { 
	val f = new FileReader("input.txt") // Use and close file 
} catch { 
	case ex: FileNotFoundException => // Handle missing file 
	case ex: IOException => // Handle other I/O error 
}

与Java的一个差别是Scala里不需要你捕获检查异常(checked exception)或把它们声明在throws子句中。如果你愿意,可以用ATthrows标注声明一个throws子句,但这不是必需的。

finally子句

没有啥要特别说明的:

import java.io.FileReader

val file = openFile()
try { 
	// ... do something ...
} finally { 
	file.close()
}

try-cache-finally产生的值

和其它大多数Scala控制结构一样,try-catch-finally也产生值。

下面的例子尝试拆分URL,但如果URL格式错误就使用缺省值。结果是,如果没有异常抛出,则对应于try子句;如果抛出异常并被捕获,则对应于相应的catch子句。如果异常被抛出但没被捕获,表达式就没有返回值。由finally子句计算得到的值,如果有的话,被抛弃。通常finally子句做一些清理类型的工作如关闭文件;他们不应该改变在主函数体或try的catch子句中计算的值。

import java.net.URL 
import java.net.MalformedURLException 

def urlFor(path: String) = 
	try { new URL(path) } 
	catch { 
		case e: MalformedURLException => 
			new URL("http://www.scalalang.org") 
}

而下面的两个例子一个第一个值为2,第二个值为1:

scala> def f(): Int = try { return 1 } finally { return 2 }
f: ()Int

scala> f()
res1: Int = 2

scala> def g(): Int = try { 1 } finally { 2 }
g: ()Int

scala> g()
res2: Int = 1

这个不明白是为什么,因此通常最好还是避免从finally子句中返回值。最好是把finally子句当作确保某些副作用,如关闭打开的文件。

match表达式

Scala的匹配表达式允许你在许多可选项(alternative)中做选择,就好象其它语言中的switch语句。通常说来match表达式可以让你使用任意的模式(pattern)做选择,后面会有专门的篇幅介绍。通用的模式可以稍等再说。目前,只要考虑使用match在若干可选项中做选择。

下面的例子里的脚本从参数列表读入食物名然后打印食物配料。match表达式检查参数列表的第一个参数firstArg。如果是字串"salt"就打印"pepper";如果是"chips",就打印"salsa",如此递推。缺省情况用下划线_说明,这是常用在Scala里作为占位符表示完全不清楚的值的通配符。

val firstArg = if (args.length > 0) args(0) else "" 

firstArg match { 
	case "salt"  => println("pepper") 
	case "chips" => println("salsa") 
	case "eggs"  => println("bacon") 
	case _       => println("huh?") 
}

与Java的switch语句比,匹配表达式还有一些重要的差别:

  • 任何种类的常量,或其他什么东西,都能用作Scala里的case,而不只是Java的case语句里面的整数类型和枚举常量。
  • 每个可选项的最后并没有break。取而代之,break是隐含的,不会有从一个可选项转到另一个里面去的情况。这通常把代码变短了,并且避免了一些错误的根源。
  • match表达式也能产生值。
val firstArg = if (args.length > 0) args(0) else "" 

val friend = firstArg match { 
	case "salt"  => "pepper"
	case "chips" => "salsa" 
	case "eggs"  => "bacon" 
	case _       => "huh?"
}
println(friend)

不要用break和contine

break和continue与函数式文本结合得不好,而且有效利用函数式文本可以让代码写得更加简短。

最简单的方式是用if替换每个every和用布尔变量替换每个break。布尔变量指代是否包含它的while循环应该继续。比如说,假设你正搜索一个参数列表去查找以“.scala”结尾但不以连号开头的字串。Java里你可以——如果你很喜欢while循环,break和continue——如此写:

  int i = 0;                // This is Java
  boolean foundIt = false;
  while (i < args.length) {
    if (args[i].startsWith("-")) {
      i = i + 1;
      continue;
    }
    if (args[i].endsWith(".scala")) {
      foundIt = true;
      break;
    }
    i = i + 1;
  }

如果要字面直译成Scala的代码,代之以执行一个if然后continue,你可以写一个if环绕while余下的全部内容。要去掉break,你可以增加一个布尔变量提示是否继续做下去,不过在这里你可以复用foundIt,基本就是这样:

  var i = 0
  var foundIt = false

  while (i < args.length && !foundIt) {
    if (!args(i).startsWith("-")) {
      if (args(i).endsWith(".scala"))
        foundIt = true
    }
    i = i + 1
  }

这个版本与原来的Java代码非常像。所有的主要段落仍然存在并保持原顺序。有两个可重新赋值的变量及一个while循环。循环内有个i是否小于args.length的测试,然后检查"-",然后检查".scala"。

如果要去掉代码里的var,可以尝试的一种方式是用递归函数重写循环。比方说,你可以定义带一个整数值做输入的searchFrom函数向前搜索,并返回想要的参数的索引。采用这种技巧的代码看上去会像这样:

  def searchFrom(i: Int): Int =
    if (i >= args.length) -1
    else if (args(i).startsWith("-")) searchFrom(i + 1) 
    else if (args(i).endsWith(".scala")) i
    else searchFrom(i + 1)

  val i = searchFrom(0)

每个continue都被带有i + 1做参数的递归调用替换掉,有效地跳转到下一个整数。用递归替代了循环的编程风格更易于理解。

Scala编译器不会实际对代码7.17展示的代码生成递归函数。因为所有的递归调用都在尾调用:tail-call位置,编译器会产生出与while循环类似的代码。每个递归调用将被实现为回到函数开始位置的跳转。尾调用优化将在后面用另外篇幅讨论。

变量作用域

大括号通常引入了一个新的范围,所以任何定义在打括号里的东西在括号之后就脱离了范围。这条规则有几个例外,因为在Scala里有时候你可以用大括号代替小括号。表达式语法的替代品是这种使用大括号例子的其中之一。

本地变量:local variable。对于它们被定义的函数来说是“本地”的。每次函数被调用的时候,一整套全新的本地变量将被使用。 一旦变量被定义了,你就不可以在同一个范围内定义同样的名字。比如,下面的脚本不会被编译通过:

  val a = 1
  val a = 2 // Does not compile
  println(a)

然而,你可以在一个内部范围内定义与外部范围里名称相同的变量。下列脚本将编译通过并可以运行。内部变量被说成是遮蔽(shadow)了同名的外部变量,因为在内部范围内外部变量变得不可见了:

  val a = 1;
  {
    val a = 2 // Compiles just fine
    println(a)
  }
  p

在解释器里看上去像是遮蔽的东西:

  scala> val a = 1
  a: Int = 1

  scala> val a = 2
  a: Int = 2

  scala> println(a)
  2

在理论上,解释器在每次你输入新的语句时都创建了一个新的嵌套范围。因此,你可以把之前解释的代码虚拟化认为是

  val a = 1;
  {
    val a = 2;
    {
      println(a)
    }
  }

重构指令式风格的代码

通过指令式风格输出乘法表:

  def printMultiTable() {

    var i = 1
    // only i in scope here

    while (i <= 10) {

      var j = 1
      // both i and j in scope here

      while (j <= 10) {

        val prod = (i * j).toString
        // i, j, and prod in scope here

        var k = prod.length
        // i, j, prod, and k in scope here

        while (k < 4) {
          print(" ")
          k += 1
        }

        print(prod)
        j += 1
      }

      // i and j still in scope; prod and k out of scope

      println()
      i += 1
    }

    // i still in scope; j, prod, and k out of scope
  }

代码在两个方面显示出了指令式风格。

首先,调用printMultiTable有副作用:在标准输出上打印乘法表。在函数式风格中,我们重构了函数,让它把乘法表作为字串返回。由于函数不再执行打印,我们把它重命名为multiTable。正如前面提到过的,没有副作用的函数的一个优点是它们很容易进行单元测试。要测试printMultiTable,你需要重定义print和println从而能够检查输出的正确性。测试multiTable就简单多了,只要检查结果即可。

  // Returns a row as a sequence
  def makeRowSeq(row: Int) =
    for (col <- 1 to 10) yield {
      val prod = (row * col).toString
      val padding = " " * (4 - prod.length)
      padding + prod
    }

  // Returns a row as a string
  def makeRow(row: Int) = makeRowSeq(row).mkString

  // Returns table as a string with one row per line
  def multiTable() = {

    val tableSeq = // a sequence of row strings
      for (row <- 1 to 10)
      yield makeRow(row)

    tableSeq.mkString("\n")
  }

printMultiTable里另一个揭露其指令式风格的信号来自于它的while循环和var。与之相对,multiTable函数使用了val,for表达式,帮助函数:helper function,并调用了mkString。

我们提炼出两个帮助函数makeRowmakeRowSeq,使代码容易阅读。

函数makeRowSeq使用for表达式从1到10枚举列数。这个for函数体计算行和列的乘积,决定乘积前占位的空格,并生成由占位空格,乘积字串叠加成的结果。for表达式的结果是一个包含了这些生成字串作为元素的序列(scala.Seq的某个子类)。

另一个帮助函数makeRow仅仅调用了makeRowSeq返回结果的mkString函数。叠加序列中的字串把它们作为一个字串返回。

multiTable方法首先使用一个for表达式的结果初始化tableSeq,这个for表达式从1到10枚举行数,对每行调用makeRow获得该行的字串。因为字串前缀yield关键字,所以表达式的结果就是行字串的序列。现在仅剩下的工作就是把字串序列转变为单一字串。mkString的调用完成这个工作,并且由于我们传递进去"\n",因此每个字串结尾插入了换行符。

函数与闭包

方法(method)

方法是被定义为某个对象成员的函数,这是最常用的形式。如下面这个工具检查文件中超过指定长度的行:

  import scala.io.Source

  object LongLines {

    def processFile(filename: String, width: Int) {
      val source = Source.fromFile(filename)
      for (line <- source.getLines) 
        processLine(filename, width, line)
    }

    private def processLine(filename: String,
        width: Int, line: String) {

      if (line.length > width)
        println(filename +": "+ line.trim)
    }
  }

再定义了一个application以后,就可以在shell中调用它了,把第一个命令行参数当作行长度,并把后续的参数解释为文件名:

  object FindLongLines {
    def main(args: Array[String]) {
      val width = args(0).toInt
      for (arg <- args.drop(1))
        LongLines.processFile(arg, width)
    } 
  } 

调用:

  $ scala FindLongLines 45 LongLines.scala
  LongLines.scala: def processFile(filename: String, width: Int) {

本地函数

Java里通过定义private方法来限制访问。在Scala里还可以把方法定义在另一个函数里来限制只有所在的代码块能访问:

  def processFile(filename: String, width: Int) {

    def processLine(filename: String,
        width: Int, line: String) {

      if (line.length > width)
        print(filename +": "+ line)
    }    

    val source = Source.fromFile(filename)
    for (line <- source.getLines) {
      processLine(filename, width, line)
    }
  }

还可以省掉filenamewidth这两个参数的传递:

  import scala.io.Source

  object LongLines {

    def processFile(filename: String, width: Int) {

      def processLine(line: String) {
        if (line.length > width)
          print(filename +": "+ line)
      }    

      val source = Source.fromFile(filename)
      for (line <- source.getLines)
        processLine(line)
    }
  }

头等函数与函数字面量(literal)

Scala拥有头等函数(first-class function),除了定义函数与调用函数外,还可以写成没有名字的函数字面量(literal)。

函数字面量直接作为一段文本被编译进一个类中,等到运行时被实例化为函数值(function value)。

任何函数值都是某个扩展了若干scala包的FunctionN特质之一的类的实例,如Function0是没有参数的函数,Function1是有一个参数的函数等等。每个FunctionN特质有一个apply方法用来调用函数。

简单例子:

(x: Int) => x + 1

=>指明这个函数把左边的东西(任何整数x)转变成右边的东西(x + 1)。所以,这是一个把任何整数x映射为x + 1的函数。

函数值是对象,所以如果你愿意可以把它们存入变量。它们也是函数,所以你可以使用通常的括号函数调用写法调用它们。以下是这两种动作的例子:

  scala> var increase = (x: Int) => x + 1
  increase: (Int) => Int = <function>

  scala> increase(10)
  res0: Int = 11

本例中,因为increase是var,你可以在之后重新赋给它不同的函数值。

  scala> increase = (x: Int) => x + 9999
  increase: (Int) => Int = <function>

  scala> increase(10)
  res2: Int = 10009

如果你想在函数文本中包括超过一个语句,用大括号包住函数体,一行放一个语句,就组成了一个代码块。与方法一样,当函数值被调用时,所有的语句将被执行,而函数的返回值就是最后一行产生的那个表达式。

  scala> increase = (x: Int) => {
       |   println("We")
       |   println("are")
       |   println("here!")
       |   x + 1
       | }
  increase: (Int) => Int = <function>

  scala> increase(10)
  We
  are
  here!
  res4: Int = 11

许多Scala库都提供了结合函数字面量的机制。例如,所有的集合类都能用到foreach方法和filter方法:

  scala> val someNumbers = List(-11, -10, -5, 0, 5, 10)
  someNumbers: List[Int] = List(-11, -10, -5, 0, 5, 10)

  scala> someNumbers.foreach((x: Int) => println(x))
  -11
  -10
  -5
  0
  5
  10

函数字面量的短格式

Scala提供了许多方法去除冗余信息并把函数文本写得更简短。

一种让函数文本更简短的方式是去除参数类型:

  scala> someNumbers.filter((x) => x > 0)
  res7: List[Int] = List(5, 10)

根据someNumbers编译器知道x一定是整数,因为它看到你立刻使用了这个函数过滤整数列表(暗示)。这被称为目标类型化(target typing),。目标类型化的精确细节并不重要。你可以简单地从编写一个不带参数类型的函数文本开始,并且,如果编译器不能识别,再加上类型。几次之后你就对什么情况编译器能或不能解开谜题有感觉了。

第二种去除无用字符的方式是省略类型是被推断的参数之外的括号。前面例子里,x两边的括号不是必须的:

  scala> someNumbers.filter(x => x > 0)
  res8: List[Int] = List(5, 10)

占位符语法(Placeholder syntax)

如果想让函数文本更简洁,可以把下划线当做一个或更多参数的占位符,只要每个参数在函数文本内仅出现一次,也就是说,在这种情况下每个下划线都代表一个不同的参数。

比如,_ > 0对于检查值是否大于零的函数来说就是非常短的标注:

  scala> someNumbers.filter(_ > 0)
  res9: List[Int] = List(5, 10)

可以把下划线看作表达式里需要被“填入”的“空白”。相当于:

  scala> someNumbers.filter(x => x > 0)
  res10: List[Int] = List(5, 10)

有时编译器有可能没有足够的信息推断缺失的参数类型。如只是写:

  scala> val f = _ + _
  <console>:4: error: missing parameter type for expanded 
  function ((x$1, x$2) => x$1.$plus(x$2))
         val f = _ + _
                 ^

这种情况下使用冒号指定类型:

  scala> val f = (_: Int) + (_: Int)
  f: (Int, Int) => Int = <function>

  scala> f(5, 10)
  res11: Int = 15

部分应用函数(partially applied function)

下划线不仅能代替一个参数,还可以代替整个参数列表:

  someNumbers.foreach(println(_))

或简化为:

  someNumbers.foreach(println _)

住要在函数名和下划线之间留一个空格,因为不这样做编译器会认为在调用名为println_的方法。

Scala把短格式直接看作是你输入了下列代码:

  someNumbers.foreach(x => println(x))

以这种方式使用下划线时,你就正在写一个偏应用函数(partially applied function)。Scala里,当你调用函数,传入任何需要的参数,你就是在把函数应用到参数上。如,给定下列函数:

  scala> def sum(a: Int, b: Int, c: Int) = a + b + c
  sum: (Int,Int,Int)Int

你就可以把函数sum应用到参数1,2和3上,如下:

  scala> sum(1, 2, 3)
  res12: Int = 6

偏应用函数是一种表达式,你不需要提供函数需要的所有参数。代之以仅提供部分,或不提供所需参数。比如,要创建不提供任何三个所需参数的调用sum的偏应用表达式,只要在“sum”之后放一个下划线即可。然后可以把得到的函数存入变量。举例如下:

  scala> val a = sum _
  a: (Int, Int, Int) => Int = <function>

有了这个代码,Scala编译器以偏应用函数表达式,sum _,实例化一个带三个缺失整数参数的函数值,并把这个新的函数值的索引赋给变量a。当你把这个新函数值应用于三个参数之上时,它就转回头调用sum,并传入这三个参数:

  scala> a(1, 2, 3)
  res13: Int = 6

实际发生的事情是这样的:

名为a的变量指向一个函数值对象。这个函数值是由Scala编译器依照偏应用函数表达式sum _,自动产生的类的一个实例。

编译器产生的类有一个apply方法带三个参数(产生的类扩展了特质Function3,定义了三个参数的apply方法)。之所以带三个参数是因为sum _表达式缺少的参数数量为三。Scala编译器把表达式a(1,2,3)翻译成对函数值的apply方法的调用,传入三个参数1,2,3。因此a(1,2,3)是下列代码的短格式:

  scala> a.apply(1, 2, 3)
  res14: Int = 6

Scala编译器根据表达式sum _自动产生的类里的apply方法,简单地把这三个缺失的参数前转到sum,并返回结果。本例中apply调用了sum(1,2,3),并返回sum返回的,6。

这种一个下划线代表全部参数列表的表达式的另一种用途,就是把它当作转换def为函数值的方式。例如,如果你有一个本地函数,如sum(a: Int, b: Int, c: Int): Int,你可以把它“包装”在apply方法具有同样的参数列表和结果类型的函数值中。当你把这个函数值应用到某些参数上时,它依次把sum应用到同样的参数,并返回结果。尽管不能把方法或嵌套函数赋值给变量,或当作参数传递给其它方法,但是如果你把方法或嵌套函数通过在名称后面加一个下划线的方式包装在函数值中,就可以做到了。

现在,尽管sum _确实是一个偏应用函数,或许对你来说为什么这么称呼并不是很明显。这个名字源自于函数未被应用于它所有的参数。在sum _的例子里,它没有应用于任何参数。不过还可以通过提供某些但不是全部需要的参数表达一个偏应用函数。举例如下:

  scala> val b = sum(1, _: Int, 3)
  b: (Int) => Int = <function>

这个例子里,你提供了第一个和最后一个参数给sum,但中间参数缺失。因为仅有一个参数缺失,Scala编译器会产生一个新的函数类,其apply方法带一个参数。在使用一个参数调用的时候,这个产生的函数的apply方法调用sum,传入1,传递给函数的参数,还有3。如下:

  scala> b(2)       // b.apply调用了sum(1,2,3)
  res15: Int = 6

  scala> b(5)       // b.apply调用了sum(1,5,3)
  res16: Int = 9

如果你正在写一个省略所有参数的偏应用程序表达式,如println _或sum _,而且在代码的那个地方正需要一个函数,你可以去掉下划线从而表达得更简明。例如,代之以打印输出someNumbers里的每一个数字:

  val someNumbers = List(-11, -10, -5, 0, 5, 10) 
  someNumbers.foreach(println _)

你可以只是写成:

  someNumbers.foreach(println) 

最后一种格式仅在需要写函数的地方,如例子中的foreach调用,才能使用。编译器知道这种情况需要一个函数,因为foreach需要一个函数作为参数传入。在不需要函数的情况下,尝试使用这种格式将引发一个编译错误。举例如下:

  scala> val c = sum
  <console>:5: error: missing arguments for method sum...
  follow this method with `_' if you want to treat it as
     a partially applied function
         val c = sum
                 ^
  scala> val d = sum _
  d: (Int, Int, Int) => Int = <function>

  scala> d(10, 20, 30)
  res17: Int = 60 

为什么要使用尾下划线? Scala的偏应用函数语法凸显了Scala与经典函数式语言如Haskell或ML之间,设计折中的差异。在经典函数式语言中,偏应用函数被当作普通的例子。更进一步,这些语言拥有非常严格的静态类型系统能够暴露出你在偏应用中可能犯的所有错误。Scala与指令式语言如Java关系近得多,在这些语言中没有应用所有参数的方法会被认为是错误的。进一步说,子类型推断的面向对象的传统和全局的根类型接受一些被经典函数式语言认为是错误的程序。

举例来说,如果你误以为List的drop(n: Int)方法如tail(),那么你会忘记你需要传递给drop一个数字。你或许会写,println(drop)。如果Scala采用偏应用函数在哪儿都OK的经典函数式传统,这个代码就将通过类型检查。然而,你会惊奇地发现这个println语句打印的输出将总是<function>!可能发生的事情是表达式drop将被看作是函数对象。因为println可以带任何类型对象,这个代码可以编译通过,但产生出乎意料的结果。

为了避免这样的情况,Scala需要你指定显示省略的函数参数,尽管标志简单到仅用一个_。Scala允许你仅在需要函数类型的地方才能省略这个仅用的_。

闭包(Closures)

函数不仅可以用到参数:

  (x: Int) => x + more  // how much more? 

,more是个自由变量:free variable,因为函数文本自身没有给出其含义。相对的,x变量是一个绑定变量:bound variable,因为它在函数的上下文中有明确意义:被定义为函数的唯一参数,一个Int。如果你尝试独立使用这个函数文本,范围内没有任何more的定义,编译器会报错说:

  scala> (x: Int) => x + more
  <console>:5: error: not found: value more
         (x: Int) => x + more
                         ^ 

另一方面,只要有一个叫做more的什么东西同样的函数文本将工作正常:

  scala> var more = 1
  more: Int = 1

  scala> val addMore = (x: Int) => x + more
  addMore: (Int) => Int = <function>

  scala> addMore(10)
  res19: Int = 11 

依照这个函数文本在运行时创建的函数值(对象)被称为闭包:closure。名称源自于通过“捕获”自由变量的绑定对函数文本执行的“关闭”行动。

不带自由变量的函数文本,如(x: Int) => x + 1,被称为封闭术语:closed term,这里术语:term指的是一小部分源代码。因此依照这个函数文本在运行时创建的函数值严格意义上来讲就不是闭包,因为(x: Int) => x + 1在编写的时候就已经封闭了。

任何带有自由变量的函数文本,如(x: Int) => x + more,都是开放术语:open term。因此,任何依照(x: Int) => x + more在运行期创建的函数值将必须捕获它的自由变量,more,的绑定。由于函数值是关闭这个开放术语(x: Int) => x + more的行动的最终产物,得到的函数值将包含一个指向捕获的more变量的参考,因此被称为闭包。

如果more在闭包创建之后被改变了闭包会反映这个变化。如下:

  scala> more = 9999
  more: Int = 9999

  scala> addMore(10)
  res21: Int = 10009 

直觉上,Scala的闭包捕获了变量本身,而不是变量指向的值。相对的,Java的内部类根本不允许你访问外围范围内可以改变的变量,因此到底是捕获了变量还是捕获了它当前具有的值就没有差别了。

反过来也同样。闭包对捕获变量作出的改变在闭包之外也可见:

  scala> val someNumbers = List(-11, -10, -5, 0, 5, 10)
  someNumbers: List[Int] = List(-11, -10, -5, 0, 5, 10)

  scala> var sum = 0
  sum: Int = 0

  scala> someNumbers.foreach(sum +=  _)

  scala> sum
  res23: Int = -11 

对于会有不同实例的场景,如:本地变量,闭包会对应到创建时关联的那个变量。

例如,以下是创建和返回“递增”闭包的函数:

  def makeIncreaser(more: Int) = (x: Int) => x + more 

每次函数被调用时都会创建一个新闭包。每个闭包都会访问闭包创建时活跃的more变量。

  scala> val inc1 = makeIncreaser(1)
  inc1: (Int) => Int = <function>

  scala> val inc9999 = makeIncreaser(9999)
  inc9999: (Int) => Int = <function> 

结果依赖于闭包被创建时more是如何定义的:

  scala> inc1(10)
  res24: Int = 11

  scala> inc9999(10)
  res25: Int = 10009 

尽管本例中more是一个已经返回的方法调用的参数也没有区别。Scala编译器在这种情况下重新安排了它以使得捕获的参数继续存在于堆中,而不是堆栈中,因此可以保留在创建它的方法调用之外。这种重新安排的工作都是自动关照的,因此你不需要操心。请任意捕获你想要的变量:val,var,或参数。

重复参数

Scala允许你指明函数的最后一个参数可以是重复的。这可以允许客户向函数传入可变长度参数列表。想要标注一个重复参数,在参数的类型之后放一个星号。例如:

  scala> def echo(args: String*) = 
       |   for (arg <- args) println(arg)
  echo: (String*)Unit 

这样定义,echo可以被零个至多个String参数调用:

  scala> echo()

  scala> echo("one")
  one

  scala> echo("hello", "world!")
  hello
  world! 

函数内部,重复参数的类型是声明参数类型的数组。因此,echo函数里被声明为类型String*的args的类型实际上是Array[String]。然而,如果你有一个合适类型的数组,并尝试把它当作重复参数传入,你会得到一个编译器错误:

  scala> val arr = Array("What's", "up", "doc?")
  arr: Array[java.lang.String] = Array(What's, up, doc?)

  scala> echo(arr)
  <console>:7: error: type mismatch;
   found   : Array[java.lang.String]
   required: String
         echo(arr)
              ^ 

要实现这个做法,你需要在数组参数后添加一个冒号和一个_*符号,像这样:

  scala> echo(arr: _*)
  What's
  up
  doc? 

这个标注告诉编译器把arr的每个元素当作参数,而不是当作单一的参数传给echo

尾递归(tail recursive)

Scala编译器可以应用一个重要的优化。注意递归调用是函数体执行的最后一件事,那么函数在它们最后一个动作调用自己的函数。这被称为尾递归。

Scala编译器检测到尾递归就用新值更新函数参数,然后把它替换成一个回到函数开头的跳转。这样减小了递归调用的开销。

递归经常是比基于循环的更优美和简明的方案。如果方案是尾递归,就无须付出任何运行期开销。

跟踪尾递归函数

尾递归函数将不会为每个调用制造新的堆栈框架;所有的调用将在一个框架内执行。所以在调试的时候会比较怪。

例如,这个函数调用自身若干次之后抛出一个异常:

  def boom(x: Int): Int = 
    if (x == 0) throw new Exception("boom!")
    else boom(x - 1) + 1 

这个函数不是尾递归,因为在递归调用之后执行了递增操作。如果执行它,你会得到预期的:

  scala>  boom(3)
  java.lang.Exception: boom!
  	at .boom(<console>:5)
  	at .boom(<console>:6)
  	at .boom(<console>:6)
  	at .boom(<console>:6)
  	at .<init>(<console>:6)
  ...

如果你现在修改了boom从而让它变成尾递归:

 def bang(x: Int): Int = 
   if (x == 0) throw new Exception("bang!")
   else bang(x - 1) 

你会得到:

  scala> bang(5)
  java.lang.Exception: bang!
  	at .bang(<console>:5)
  	at .<init>(<console>:6)
  ... 

这回,你仅看到了bang的一个堆栈框架。或许你会认为bang在调用自己之前就崩溃了,但这不是事实。如果你认为你会在看到堆栈跟踪时被尾调用优化搞糊涂,你可以用开关项关掉它:

  -g:notailcalls 

把这个参数传给scala的shell或者scalac编译器。定义了这个选项,你就能得到一个长长的堆栈跟踪了:

  scala> bang(5)
  java.lang.Exception: bang!
  	at .bang(<console>:5)
  	at .bang(<console>:5)
  	at .bang(<console>:5)
  	at .bang(<console>:5)
  	at .bang(<console>:5)
  	at .bang(<console>:5)
  	at .<init>(<console>:6)
  ... 

尾递归的局限

Scala里尾递归的使用局限很大,因为JVM指令集使实现更加先进的尾递归形式变得很困难。Scala仅优化了直接递归调用使其返回同一个函数。如果递归是间接的,就像在下面的例子里两个互相递归的函数,就没有优化的可能性了:

  def isEven(x: Int): Boolean =
    if (x == 0) true else isOdd(x - 1)
  def isOdd(x: Int): Boolean =
    if (x == 0) false else isEven(x - 1) 

同样如果最后一个调用是一个函数值你也不能获得尾调用优化。请考虑下列递归代码的实例:

  val funValue = nestedFun _
  def nestedFun(x: Int) { 
    if (x != 0) { println(x); funValue(x - 1) }
  } 

funValue变量指向一个实质是包装了nestedFun的调用的函数值。当你把这个函数值应用到参数上,它会转向把nestedFun应用到同一个参数,并返回结果。因此你或许希望Scala编译器能执行尾调用优化,但在这个例子里做不到。因此,尾调用优化受限于方法或嵌套函数在最后一个操作调用本身,而没有转到某个函数值或什么其它的中间函数的情况。

控制抽象

可复用的代码

所有的函数都被分割成通用部分(它们在每次函数调用中都相同)以及非通用部分(在不同的函数调用中可能会变化)。通用部分是函数体,而非通用部分必须由参数提供。

当你把函数值用做参数时,算法的非通用部分就是它代表的某些其它算法。在这种函数的每一次调用中,你都可以把不同的函数值作为参数传入,于是被调用函数将在每次选用参数的时候调用传入的函数值。这种高阶函数(higher-order function)带其它函数做参数的函数提供了机会去组织和简化代码。

例子。一个工具类,提供了很多查找文件的方法,有根据文件结尾的、文件名是否包含指定字串的、文件名是否匹配正则的:

  object FileMatcher {

    // private method, get file name list in current dir
    private def filesHere = (new java.io.File(".")).listFiles

    // by file name end with string
    def filesEnding(query: String) =
      for (file <- filesHere; if file.getName.endsWith(query))
        yield file
                     
    // by file name end include string
    def filesContaining(query: String) =
      for (file <- filesHere; if file.getName.contains(query))
        yield file
                   
    // by file name match regex
    def filesRegex(query: String) =
      for (file <- filesHere; if file.getName.matches(query))
        yield file
  } 

如果在Java中对应这种情况,大家应该都知道如何提炼接口来重用代码,这里就不啰嗦了。

如果是在某些动态语言中,要提炼一个工具方法提炼出共用的部分,根据传入不同method作为参数也匹配也很方便,可以直接把代码“拼接”起来:

  def filesMatching(query: String, method) =
    for (file <- filesHere; if file.getName.method(query))
      yield file

不过Scala不是动态语言,不能这么拼接。虽然不能把方法名作为参数传递,但可以通过字面量在运行时产生对应的函数值:

  def filesMatching(
    query: String,
    matcher: (String, String) => Boolean
  ) = {
    for (file <- filesHere; if matcher(file.getName, query))
      yield file
  } 

字面量只说明了函数的类型是(String, String) => Boolean,不用关内部逻辑的。现在已经有了一个filesMatching方法来处理共同的逻辑,三个具体的匹配方法只要调用它就行了:

  def filesEnding(query: String) =
    filesMatching(query, _.endsWith(_))

  def filesContaining(query: String) =
    filesMatching(query, _.contains(_))

  def filesRegex(query: String) =
    filesMatching(query, _.matches(_)) 

加上参数表和参数类型可以更加好理解一些

  // _.endsWith(_)
  (fileName: String, query: String) => fileName.endsWith(query)
  
  // _.contains(_)
  (fileName: String, query: String) => fileName.contains(query))
  
  // _.matches(_)
  (fileName: String, query: String) => fileName.matches(query))

代码已经被简化了,但它实际还能更短。注意到query传递给了方法filesMatching,但filesMatching根本用不着这个参数,只是为了把它传回给传入的matcher函数。

所以在这里可以直接把参数query绑定到函数字面量中,这样fileMacthing方法就不要query这个参数了。

  object FileMatcher {
    private def filesHere = (new java.io.File(".")).listFiles

    private def filesMatching(matcher: String => Boolean) =
      for (file <- filesHere; if matcher(file.getName))
        yield file
  
    def filesEnding(query: String) =
      filesMatching(_.endsWith(query))
  
    def filesContaining(query: String) =
      filesMatching(_.contains(query))
  
    def filesRegex(query: String) =
      filesMatching(_.matches(query))
  } 

简化客户端代码

高阶函数可以提供更加强大的API,让客户的代码写起来更加简单。

比如List中的高阶函数exists方法已经提供了遍历整个集合的抽象,用户只要把判断符合的函数传入就可以了。下面的两个例子非常简单地实现了检查是否存在负数和是否存在奇数两个方法:

scala> def containsNeg(nums: List[Int]) = nums.exists(_ < 0)
containsNeg: (nums: List[Int])Boolean

scala> def containsOdd(nums: List[Int]) = nums.exists(_ % 2 == 1)
containsOdd: (nums: List[Int])Boolean
 
scala> List(1, 2, 3, 4)
res1: List[Int] = List(1, 2, 3, 4)

scala> containsNeg(res1)
res3: Boolean = false

scala> containsOdd(res1)
res4: Boolean = true

如果没有高阶函数exists,那就要自己写循环的逻辑,就会有很多重复的代码:

  def containsNeg(nums: List[Int]): Boolean = {
    var exists = false
    for (num <- nums)
      if (num < 0)
        exists = true
    exists
  }
  
  def containsOdd(nums: List[Int]): Boolean = {
    var exists = false
    for (num <- nums)
      if (num % 2 == 1)
        exists = true
    exists
  } 

柯里化(Currying)

理解柯里化可以帮助理解如何建立自己的控制结构。柯里化就是一个函数有多个参数列表。

普通的函数,实现了两个Int型参数,x和y的加法:

  scala> def plainOldSum(x: Int, y: Int) = x + y
  plainOldSum: (Int,Int)Int

  scala> plainOldSum(1, 2)
  res4: Int = 3 

curry化后的同一个函数,两个列表的各一个参数:

  scala> def curriedSum(x: Int)(y: Int) = x + y
  curriedSum: (Int)(Int)Int

  scala> curriedSum(1)(2)
  res5: Int = 3 

实际上背靠背地调用了两个传统函数。第一个函数调用带单个的名为x的Int参数,并返回第二个函数的函数值。第二个函数带Int参数y。下面的名为first的函数实质上执行了curriedSum的第一个传统函数调用会做的事情:

  scala> def first(x: Int) = (y: Int) => x + y
  first: (Int)(Int) => Int 

调用第一个函数并传入1——会产生第二个函数:

  scala> val second = first(1)
  second: (Int) => Int = <function> 

调用第二个函数传入2产生结果:

  scala> second(2)
  res6: Int = 3 

first和second函数只演示连接在curriedSum函数上的那两个函数,并不直接连接在curriedSum函数上的那两个函数。但我们仍然有一个方式获得实际指向curriedSum的“第二个”函数的参考。你可以用偏应用函数表达式方式,把占位符标注用在curriedSum里,如:

  scala> val onePlus = curriedSum(1)_
  onePlus: (Int) => Int = <function> 

之前说过,当占位符标注用在传统方法上时,如println _,你必须在名称和下划线之间留一个空格。不然编译器会误认为是要调用名为println_的函数。而在这个例子里不需要,因为println_是Scala里合法的标识符,curriedSum(1)_不是。

现在得到了指向一个函数的引用,这个函数在被调用的时候,传入Int参数加一并返回结果:

  scala> onePlus(2)
  res7: Int = 3 

由于第二个函数的参数已经有了(传入的是2),现在再用参数2调用第一个函数也能有结果出来:

  scala> val twoPlus = curriedSum(2)_
  twoPlus: (Int) => Int = <function>

  scala> twoPlus(2)
  res8: Int = 4 

编写新的控制结构

在拥有头等函数的编程语言中,可以在方法中以函数作为参数创造自己的控制结构。

比如有个“重复操作”的方法,它可以把任何操作重复执行两次:

  scala> def twice(op: Double => Double, x: Double) = op(op(x))
  twice: ((Double) => Double,Double)Double

  scala> twice(_ + 1, 5)
  res9: Double = 7.0 

如果在工作中曾经遇到重复操作两次(Double) => Double类型函数的操作的话,这样就把一个控制结构给抽象出来了。

再考虑一个常用的工作流程:打开一个资源,对它进行操作,然后关闭资源。你可以使用如下的方法将其捕获并放入控制抽象:

  def withPrintWriter(file: File, op: PrintWriter => Unit) {
    val writer = new PrintWriter(file)
    try {
      op(writer)
    } finally {
      writer.close()
    }
  } 

以后要使用的时候就只要传入要处理的文件和处理的方法就行了,打开一个资源和关闭资源都已经在高阶函数中被抽象出来了:

  withPrintWriter(
    new File("date.txt"),
    writer => writer.println(new java.util.Date)
  ) 

这个技巧被称为贷出模式:loan pattern,因为控制抽象函数,如withPrintWriter,打开了资源并“贷出”给函数。当函数完成的时候,它发出信号说明它不再需要“借”的资源。于是资源被关闭在finally块中,以确信其确实被关闭,而忽略函数是正常结束返回还是抛出了异常。

让客户代码看上去更像内建控制结构的一种方式是使用大括号代替小括号包围参数列表。Scala的任何方法调用,如果你确实只传入一个参数,就能可选地使用大括号替代小括号包围参数:

  scala> println("Hello, world!")
  Hello, world!
  
  scala> println { "Hello, world!" }
  Hello, world!  

这个大括号技巧仅在你传入一个参数时有效,多个参数只能用小括号:

  scala> val g = "Hello, world!"
  g: java.lang.String = Hello, world!
  
  scala> g.substring(7, 9)
  res12: java.lang.String = wo 

  scala> g.substring { 7, 9 }
  <console>:1: error: ';' expected but ',' found.
         g.substring { 7, 9 }
                        ^ 

以前面例子里定义的withPrintWriter方法举例。在它最近的形式里,withPrintWriter带了两个参数,因此你不能使用大括号。虽然如此,因为传递给withPrintWriter的函数是列表的最后一个参数,你可以使用curry化把第一个参数,File拖入分离的参数列表。这将使函数仅剩下列表的第二个参数作为唯一的参数:

  def withPrintWriter(file: File)(op: PrintWriter => Unit) {
    val writer = new PrintWriter(file)
    try {
      op(writer)
    } finally {
      writer.close()
    }
  } 

可以用更赏心悦目的语法格式调用这个方法:

  val file = new File("date.txt")

  withPrintWriter(file) {
    writer => writer.println(new java.util.Date)
  } 

第一个参数列表,包含了一个File参数,被写成包围在小括号中。第二个参数列表,包含了一个函数参数,被包围在大括号中。

传名参数

上节展示的withPrintWriter方法不同于语言的内建控制结构,如if和while,在于大括号之间的代码带了参数。withPrintWriter方法需要一个类型为PrintWriter的参数。这个参数以“writer =>”方式显示出来:

  withPrintWriter(file) {
    writer => writer.println(new java.util.Date)
  } 

然而如果你想要实现某些更像if或while的东西,根本没有值要传入大括号之间的代码,那该怎么做呢?为了解决这种情况,Scala提供了叫名参数。

为了举一个有现实意义的例子:虽然Scala提供了它自己的assert,但是用户想自己实现一个称为myAssert的断言架构。

myAssert函数将带一个函数值做输入并参考一个标志位来决定该做什么。如果标志位被设置了,myAssert将调用传入的函数并证实其返回true。如果标志位被关闭了,myAssert将安静地什么都不做。 如果没有叫名参数,你可以这样写myAssert:

  var assertionsEnabled = true

  def myAssert(predicate: () => Boolean) =
    if (assertionsEnabled && !predicate())
      throw new AssertionError 

调用时不能省略() =>

  myAssert(() => 5 > 3) 

  myAssert(5 > 3) // Won't work, because missing () =>  

传名函数恰好为了实现你的愿望而出现。要实现一个叫传函数,要定义参数的类型开始于=>而不是() =>。如,改() => Boolean=> Boolean

  def byNameAssert(predicate: => Boolean) =
    if (assertionsEnabled && !predicate)
      throw new AssertionError 

现在可以省略了,看起来像语言内建的控制结构一样:

  byNameAssert(5 > 3) 

传名类型中,空的参数列表()被省略,它仅在参数中被允许。没有什么叫名变量或叫名字段这样的东西。

对于myAssert,我们费了这么大的力气,只是为了让函数字面量看起来像表达式,那为什么不直接用Boolean变量作为参数呢?

  def boolAssert(predicate: Boolean) =
    if (assertionsEnabled && !predicate)
      throw new AssertionError 

当然这种格式同样合法,并且使用这个版本boolAssert的代码看上去仍然与前面的一样:

  boolAssert(5 > 3) 

虽然如此,这两种方式之间存在一个非常重要的差别须指出:表达式会在传入参数前先被执行。

所以在上面的例子中,如果断言被禁用,你会看到boolAssert括号里的表达式的某些副作用,而byNameAssert却没有。例如,如果断言被禁用,boolAssert的例子里尝试对“x / 0 == 0”的断言将产生一个异常:

  scala> var assertionsEnabled = false
  assertionsEnabled: Boolean = false

  scala> boolAssert(x / 0 == 0)
  java.lang.ArithmeticException: / by zero
  	   at .<init>(<console>:8)
          at .<clinit>(<console>)
          at RequestResult$.<init>(<console>:3)
          at RequestResult$.<clinit>(<console>)...

但在byNameAssert的例子里尝试同样代码的断言将不产生异常:

  scala> byNameAssert(x / 0 == 0) 

组合与继承

定制一个二维布局库

作为本章运行的例子,我们将创造一个制造和渲染二维布局元素的库。每个元素将代表一个填充字符的长方形。方便起见,库将提供名为elem的工厂方法来通过传入的数据构造新的元素。例如,你将能通过工厂方法采用下面的写法创建带有字串的元素:

  elem(s: String): Element 

元素将以名为Element的类型为模型。你将能在元素上调用above或beside,把另一个元素放在当前元素的右边或是上边:

  val column1 = elem("hello") above elem("***")
  val column2 = elem("***") above elem("world")
  column1 beside column2

打印这个表达式的结果将是:

  hello ***  
   *** world

抽象类

布局元素名为Element,存放的文本内容类型为Array[String]。提供方法contents取得存放的文本内容,但没有定义实现方式,所以这个类是抽象类,要加上abstract关键字:

  abstract class Element {
    def contents: Array[String]
  } 

请注意类Element的contents方法并没带有abstract修饰符。不像Java,方法的声明中不需要(或允许)抽象修饰符。如果方法没有实现,它就是抽象的。

另一个术语用法需要分辨声明(declaration)和定义(definition)。类Element声明了抽象方法contents,但当前没有定义具体方法。

定义无参数方法

添加显示宽度和高度的方法:

height方法返回contents里的行数。

width方法返回第一行的长度,或如果元素没有行记录,返回零。(也就是说你不能定义一个高度为零但宽度不为零的元素。)

  abstract class Element {
    def contents: Array[String]
    def height: Int = contents.length
    def width: Int = if (height == 0) 0 else contents(0).length
  } 

三个方法没一个有参数列表,甚至连个空列表都没有。如:

  def width(): Int
  // 省略括号
  def width: Int 

推荐的惯例是在没有参数并且方法仅通过读含有对象的方式访问可变状态(专指其不改变可变状态)时使用无参数方法。这样感觉上就和只读字段一样,其实也可以选择把width和height作为字段而不是方法来实现,只要简单地在每个实现里把def修改成val即可:

  abstract class Element {
    def contents: Array[String]
    val height = contents.length
    val width = 
      if (height == 0) 0 else contents(0).length
  }

两组定义从客户的观点来看是完全相同的。唯一的差别是与的访问或许稍微比方法调用要快,因为字段值在类被初始化的时候被预计算,而方法调用在每次调用的时候都要计算。换句话说,字段在每个Element对象上需要更多的内存空间。因此类的使用概况,属性表达成字段还是方法更好,决定了其实现,并且这个概况还可以随时改变。

重点是Element类的客户不应在其内部实现改变的时候受影响。

特别是如果类的字段变成了访问函数,且访问函数是纯的,就是说它没有副作用并且不依赖于可变状态,那么类Element的客户不需要被重写。客户都不应该需要关心这些。

目前为止一切良好。但仍然有些琐碎的复杂的东西要去做以协同Java处理事情的方式。问题在于Java没有实现统一访问原则。因此Java里是string.length(),不是string.length(尽管是array.length,不是array.length())。不用说,这让人很困惑。

为了在这道缺口上架一座桥梁,Scala在遇到混合了无参数和空括号方法的情况时很大度。特别是,你可以用空括号方法重载无参数方法,并且反之亦可。你还可以在调用任何不带参数的方法时省略空的括号。例如,下面两行在Scala里都是合法的:

  Array(1, 2, 3).toString
  "abc".length

原则上Scala的函数调用中可以省略所有的空括号。然而,在调用的方法表达的超过其接收调用者对象的属性时,推荐仍然写一对空的括号。例如,如果方法执行了I/O,或写入可重新赋值的变量(var),或读出不是接受调用者的字段的var,无论是直接的还是非直接的通过使用可变对象,那么空括号是合适的。这种方式是让参数列表扮演一个可见的线索说明某些有趣的计算正通过调用被触发。例如:

  "hello".length  // no () because no side-effect
  println()       // better to not drop the ()

总结起来,Scala里定义不带参数也没有副作用的方法为无参数方法,也就是说,省略空的括号,是鼓励的风格。另一方面,永远不要定义没有括号的带副作用的方法,因为那样的话方法调用看上去会像选择一个字段。这样你的客户看到了副作用会很奇怪。相同地,当你调用带副作用的函数,请确信写这个调用的时候包括了空的括号。另一种考虑这个问题的方式是,如果你调用的函数执行了操作,使用括号,但如果仅提供了对某个属性的访问,省略括号。

扩展类

实例化一个元素,我们需要创建扩展了Element并实现抽象的contents方法的子类。

  class ArrayElement(conts: Array[String]) extends Element {
    def contents: Array[String] = conts
  }

这种extends子句有两个效果:使类ArrayElement从类Element继承所有非私有的成员,并且使ArrayElement成为Element的子类型。由于ArrayElement扩展了Element,类ArrayElement被称为类Element的子类。反过来,Element是ArrayElement的超类。

如果你省略extends子句,Scala编译器隐式地假设你的类扩展自scala.AnyRef,在Java平台上与java.lang.Object一致。因此,类Element隐式地扩展了类AnyRef。

ArrayElement的contents方法重载(或者可说成:实现)了类Element的抽象方法contents:

  scala> val ae = new ArrayElement(Array("hello", "world"))
  ae: ArrayElement = ArrayElement@d94e60

  scala> ae.width
  res1: Int = 5

子类型化:subtyping是指子类的值可以被用在需要其超类的值的任何地方。例如:

  val e: Element = new ArrayElement(Array("hello"))

重载方法和字段

字段和方法属于相同的命名空间。这使得字段重载无参数方法成为可能。比如说,你可以改变类ArrayElement中contents的实现,从一个方法变为一个字段,而无需修改类Element中contents的抽象方法定义:

  class ArrayElement(conts: Array[String]) extends Element {
    val contents: Array[String] = conts
  }

这个ArrayElement的版本里,字段contents(用val定义)完美地实现了类Element里的无参数方法contents(用def定义)。

另一方面,Scala里禁止在同一个类里用同样的名称定义字段和方法,而在Java里这样做被允许。例如,下面的Java类能够很好地编译:

  // This is Java
  class CompilesFine {
    private int f = 0;
    public int f() {
      return 1;
    }
  }

但是相应的Scala类将不能编译:


  class WontCompile {
    private var f = 0 // Won't compile, because a field 
    def f = 1         // and method have the same name
  }

Java为定义准备了四个命名空间:字段,方法,类型和包。

而Scala仅有两个,与Java的四个命名空间相对:

  • 值(字段,方法,包还有单例对象)
  • 类型(类和特质名)

Scala把字段和方法放进同一个命名空间的理由很清楚,因为这样你就可以使用val重载无参数的方法,这种你在Java里做不到的事情。

定义参数化字段

ArrayElement类的定义。它有一个参数conts,其唯一目的是被复制到contents字段。选择conts这个参数的名称只是为了让它看上去更像字段名contents而不会与它发生实际冲突。这是一种“代码异味”,一个表明或许某些不必须的累赘和重复。

可以通过在单一的参数化字段:parametric field定义中组合参数和字段避免:

  class ArrayElement(val contents: Array[String]) extends Element

现在拥有一个可以从类外部访问的,(不能重新赋值的)字段contents。字段使用参数值初始化。等同于:

  class ArrayElement(x123: Array[String]) extends Element { 
    val contents: Array[String] = x123
  } 

同样也可以使用var前缀类参数,这种情况下相应的字段将能重新被赋值。还有可能添加如private,protected,或override这类的修饰符到这些参数化字段上,就好象你可以在其他类成员上做的事情:

  class Cat {
    val dangerous = false
  }
  class Tiger(
    override val dangerous: Boolean,
    private var age: Int
  ) extends Cat

Tiger的定义是以下包括重载成员dangerous和private成员age的类定义替代写法的简写:

  class Tiger(param1: Boolean, param2: Int) extends Cat {
    override val dangerous = param1
    private var age = param2
  }

调用超类构造器

如果再要新的子类:

  class LineElement(s: String) extends ArrayElement(Array(s)) {
    override def width = s.length
    override def height = 1
  }

由于LineElement扩展了ArrayElement,并且ArrayElement的构造器带一个参数(Array[String]),LineElement需要传递一个参数到它的超类的主构造器。要调用超类构造器,只要把你要传递的参数或参数列表放在超类名之后的括号里即可。

使用override修饰符

考虑一下这样的场景:

基类和子类是不同的人维护的。原来基类里没有add方法,所以子类里加上了。后来基类里也加上了add方法,但维护子类的人不知道。这样的规定是为了防止“脆基类”问题。

所以Scala里override有强制的规定:

  • 如果实现了抽象成员,加不加随便。
  • 如果重载了具体实现,就一定要加。
  • 没有重载就绝不能加。

这样起码保证了维护子类的人知道自己会覆盖超类的方法。

多态和动态绑定

创建一个新的子类,它可以按给出的长度宽度,用指定的字符填充:

  class UniformElement(
    ch: Char, 
    override val width: Int,
    override val height: Int 
  ) extends Element {
    private val line = ch.toString * width
    def contents = Array.make(height, line)
  }

父类的变量可以存放子类的实例,就是多态的一种体现。这么多子类都可以用父类的变量来存放:

  val e1: Element = new ArrayElement(Array("hello", "world"))
  val ae: ArrayElement = new LineElement("hello")
  val e2: Element = ae
  val e3: Element = new UniformElement('x', 2, 3)

变量和表达式上的方法调用是动态绑定(dynamically bound)的。这意味着被调用的实际方法实现取决于运行期对象其实的类,而不是变量或表达式的类型。

为了演示这种行为,我们会从我们的Element类中临时移除所有存在的成员并添加一个名为demo的方法。我们会在ArrayElement和LineElement中重载demo,但UniformElement除外:

  abstract class Element {
    def demo() {
      println("Element's implementation invoked")
    }
  }

  class ArrayElement extends Element {
    override def demo() {
      println("ArrayElement's implementation invoked")
    }
  }

  class LineElement extends ArrayElement {
    override def demo() {
      println("LineElement's implementation invoked")
    }
  }

  // UniformElement inherits Element's demo
  class UniformElement extends Element 

如果你把这些代码输入到了解释器中,那么你就能定义这个带了一个Element并调用demo的方法:

  def invokeDemo(e: Element) {
    e.demo()
  }

如果你传给invokeDemo一个ArrayElement,你会看到一条消息指明ArrayElement的demo实现被调用,尽管被调用demo的变量e的类型是Element:

  scala> invokeDemo(new ArrayElement)
  ArrayElement's implementation invoked

相同的,如果你传递LineElement给invokeDemo,你会看到一条指明LineElement的demo实现被调用的消息:

  scala> invokeDemo(new LineElement)
  LineElement's implementation invoked

传递UniformElement时的行为一眼看上去会有些可以,但是正确:

  scala> invokeDemo(new UniformElement)
  Element's implementation invoked

因为UniformElement没有重载demo,它从它的超类Element继承了demo的实现。因此,当对象的类是UniformElement时,Element的实现就是要调用的demo的正确实现。

定义final成员

要确保成员不被子类重载。Scala里和Java里一样,通过添加final修饰符给成员来做到。

  class ArrayElement extends Element {
    final override def demo() {
      println("ArrayElement's implementation invoked")
    }
  }

在类的声明上添加final修饰符把整个类声明为final:

  final class ArrayElement extends Element {
    override def demo() {
      println("ArrayElement's implementation invoked")
    }
  }

使用组合与继承

组合与继承是利用其它现存类定义新类的两个方法。

如果你接下来的工作主要是代码重用,通常你应采用组合而不是继承。只有继承受脆基类问题之苦,这 种情况你可能会无意中通过改变超类而破坏了子类。

关于继承关系你可以问自己一个问题,是否它建模了一个is-a关系。你能问的另一个问题是,是否客户 想要把子类类型当作超类类型来用。

实现示例中的功能

把一个元素放在另一个上面是指串连这两个元素的contents值。

  def above(that: Element): Element =
    new ArrayElement(this.contents ++ that.contents)

把两个元素靠在一起,我们将创造一个新的元素,其中的每一行都来自于两个元素的相应行的串连。

  def beside(that: Element): Element = {
    val contents = new Array[String](this.contents.length)
    for (i <- 0 until this.contents.length) 
      contents(i) = this.contents(i) + that.contents(i)
    new ArrayElement(contents)
  }

索引数组的循环是指令式风格。这个方法可以替代缩减成一个表达式:

  new ArrayElement(
    for (
      (line1, line2) <- this.contents zip that.contents
    ) yield line1 + line2
  )

zip操作符转换为一个对子的数组(可以称为Tupele2)。zip方法从它的两个参数中拣出相应的元素并组 织成对子数组。

例如,表达式:

  Array(1, 2, 3) zip Array("a", "b")

将生成:

  Array((1, "a"), (2, "b"))

如果两个操作数组的其中一个比另一个长,zip将舍弃余下的元素。

定义toString方法返回元素格式化成的字串:

  override def toString = contents mkString "\n"

最后是这个样子:

  abstract class Element {

    def contents: Array[String]

    def width: Int =
      if (height == 0) 0 else contents(0).length

    def height: Int = contents.length

    def above(that: Element): Element =
      new ArrayElement(this.contents ++ that.contents)

    def beside(that: Element): Element =
      new ArrayElement(
        for (
          (line1, line2) <- this.contents zip that.contents
        ) yield line1 + line2
      )

    override def toString = contents mkString "\n"
  }

定义工厂对象

最直接的方案是创建类Element的伴生对象并把它做成布局元素的工厂方法。这种方式唯一要暴露给客户 的就是Element的类/对象组合,隐藏它的三个实现类ArrayElement,LineElement和UniformElement。

  object Element {

    def elem(contents: Array[String]): Element = 
      new ArrayElement(contents)

    def elem(chr: Char, width: Int, height: Int): Element = 
      new UniformElement(chr, width, height)

    def elem(line: String): Element = 
      new LineElement(line)
  }

这些工厂方法使得改变类Element的实现通过使用elem工厂方法实现而不用显式地创建新的ArrayElement 实例成为可能。

为了不使用单例对象的名称,Element,认证而调用工厂方法,我们将在源文件顶上引用Element.elem。 换句话说,代之以在Element类内部使用Element.elem调用工厂方法,我们将引用Element.elem,这样我 们只要使用它们的简化名,elem,就可以调用工厂方法。

  import Element.elem

  abstract class Element {

    def contents: Array[String]

    def width: Int =
      if (height == 0) 0 else contents(0).length

    def height: Int = contents.length

    def above(that: Element): Element =
      elem(this.contents ++ that.contents)

    def beside(that: Element): Element =
      elem(
        for (
          (line1, line2) <- this.contents zip that.contents
        ) yield line1 + line2
      )

    override def toString = contents mkString "\n"
  }

有了工厂方法之后,子类ArrayElement,LineElement和UniformElement不再需要直接被客户访问,所以可以改成是私有的。

Scala里,你可以在类和单例对象中定义其它的类和单例对象。因此一种让Element的子类私有化的方式 就是把它们放在Element单例对象中并在那里声明它们为私有。需要的时候,这些类将仍然能被三个elem 工厂方法访问。


    private class ArrayElement(
      val contents: Array[String]
    ) extends Element

    private class LineElement(s: String) extends Element {
      val contents = Array(s)
      override def width = s.length
      override def height = 1
    }

    private class UniformElement(
      ch: Char,
      override val width: Int,
      override val height: Int
    ) extends Element {
      private val line = ch.toString * width
      def contents = Array.make(height, line)
    }

    def elem(contents:  Array[String]): Element =
      new ArrayElement(contents)

    def elem(chr: Char, width: Int, height: Int): Element =
      new UniformElement(chr, width, height)

    def elem(line: String): Element =
      new LineElement(line)
  }

变高变宽

Element的版本并不完全,因为他不允许客户把不同宽度的元素堆叠在一起,或者不同高度的元素靠在一起。比方说,下面的表达式将不能正常工作,因为组合元素的第二行比第一行要长:

  new ArrayElement(Array("hello")) above 
  new ArrayElement(Array("world!"))

与之相似的,下面的表达式也不能正常工作:


  new ArrayElement(Array("one", "two")) beside 
  new ArrayElement(Array("one"))

添加私有帮助方法widen通过带个宽度做参数并返回那个宽度的Element。heighten,能在竖直方向执行同样的功能。

  import Element.elem

  abstract class Element {
    def contents:  Array[String]

    def width: Int = contents(0).length
    def height: Int = contents.length

    def above(that: Element): Element = {
      val this1 = this widen that.width
      val that1 = that widen this.width
      elem(this1.contents ++ that1.contents)
    }

    def beside(that: Element): Element = {
      val this1 = this heighten that.height
      val that1 = that heighten this.height
      elem(
        for ((line1, line2) <- this1.contents zip that1.contents) 
        yield line1 + line2)
    }

    def widen(w: Int): Element = 
      if (w <= width) this
      else {
        val left = elem(' ', (w - width) / 2, height) 
        var right = elem(' ', w - width - left.width, height)
        left beside this beside right
      }

    def heighten(h: Int): Element = 
      if (h <= height) this
      else {
        val top = elem(' ', width, (h - height) / 2)
        var bot = elem(' ', width, h - height - top.height)
        top above this above bot
      }

    override def toString = contents mkString "\n"
  }

完整的示例代码

写一个画给定数量边界的螺旋的程序。

// In file compo-inherit/Spiral.scala

  import Element.elem

  object Spiral {

    val space = elem(" ")
    val corner = elem("+")

    def spiral(nEdges: Int, direction: Int): Element = {
      if (nEdges == 1)
        elem("+")
      else {
        val sp = spiral(nEdges - 1, (direction + 3) % 4)
        def verticalBar = elem('|', 1, sp.height)
        def horizontalBar = elem('-', sp.width, 1)
        if (direction == 0)
          (corner beside horizontalBar) above (sp beside space)
        else if (direction == 1)
          (sp above space) beside (corner above verticalBar)
        else if (direction == 2)
          (space beside sp) above (horizontalBar beside corner)
        else
          (verticalBar above corner) beside (space above sp)
      }
    }

    def main(args: Array[String]) {
      val nSides = args(0).toInt
      println(spiral(nSides, 0))
    }
  }
$ scala Spiral 6    $ scala Spiral 11    $ scala Spiral 17
+-----              +----------          +----------------
|                   |                    |                
| +-+               | +------+           | +------------+ 
| + |               | |      |           | |            | 
|   |               | | +--+ |           | | +--------+ | 
+---+               | | |  | |           | | |        | | 
                    | | ++ | |           | | | +----+ | | 
                    | |    | |           | | | |    | | | 
                    | +----+ |           | | | | ++ | | | 
                    |        |           | | | |  | | | | 
                    +--------+           | | | +--+ | | | 
                                         | | |      | | | 
                                         | | +------+ | | 
                                         | |          | | 
                                         | +----------+ | 
                                         |              | 
                                         +--------------+ 

Scala类的层级

Scala里,每个类都继承自通用的名为Any的超类。因为所有的类都是Any的子类,那么定义在Any中的方 法就是“普遍”方法:它们可以被任何对象调用。

Scala还在层级的底端定义了Null和Nothing,主要都扮演通用的子类。例如,就像说Any是所有其它类的 超类,Nothing是所有其它类的子类。

scala.hirtc

Scala类的概览

层级的顶端是类Any,定义了包含下列的方法:

  final def ==(that: Any): Boolean
  final def !=(that: Any): Boolean
  def equals(that: Any): Boolean
  def hashCode: Int
  def toString: String

类Any里的=!=,被声明为final,因此它们不能在子类里面重载。实际上,==总是与equals相同,!=总是与equals相反。因此独立的类可以通过重载equals方法修改==!=的意义。

根类Any有两个子类:AnyVal和AnyRef。

值类型(AnyVal)

AnyVal是Scala里每个内建值类型的父类。有九个这样的值类型:Byte,Short,Char,Int,Long,Float,Double,Boolean和Unit。其中的前八个对应到Java的原始类型,它们的值在运行时表示成Java的原始值。

Scala里这些类的实例都写成字面量,不能使用new创造这些类的实例。值类都被定义为即是抽象的又是final的,强制贯彻。因此如果你写了new就会出错:

  scala> new Int
  <console>:5: error: class Int is abstract; cannot be 
  instantiated
         new Int
         ^

另一个值类型,Unit,大约对应于Java的void类型;被用作不返回任何有趣结果的方法的结果类型。Unit只有一个实例值,被写作()。

值类型支持作为方法的通用的数学和布尔操作符。例如,Int有名为+*的方法,Boolean有名为||&&的方法。值类型也从类Any继承所有的方法:

  scala> 42 max 43
  res4: Int = 43

  scala> 42 min 43
  res5: Int = 42

  scala> 1 until 5
  res6: Range = Range(1, 2, 3, 4)

  scala> 1 to 5
  res7: Range.Inclusive = Range(1, 2, 3, 4, 5)

  scala> 3.abs
  res8: Int = 3

  scala> (-3).abs
  res9: Int = 3

值类型的空间是扁平的;所有的值类都是scala.AnyVal的子类型,但是它们不是互相的子类。代之以它们不同的值类型之间可以隐式地互相转换。例如,需要的时候,类scala.Int的实例可以自动放宽(通过隐式转换)到类scala.Long的实例。

隐式转换还用来为值类型添加更多的功能。例如,类型Int支持以下所有的操作:

  scala> 42 max 43
  res4: Int = 43

  scala> 42 min 43
  res5: Int = 42

  scala> 1 until 5
  res6: Range = Range(1, 2, 3, 4)

  scala> 1 to 5
  res7: Range.Inclusive = Range(1, 2, 3, 4, 5)

  scala> 3.abs
  res8: Int = 3

  scala> (-3).abs
  res9: Int = 3

工作原理:方法min,max,until,to和abs都定义在类scala.runtime.RichInt里,并且有一个从类Int到RichInt的隐式转换。当你在Int上调用没有定义在Int上但定义在RichInt上的方法时,这个转换就被应用了:

引用类型(AnyRef)

类Any的另一个子类是类AnyRef。这个是Scala里所有引用类的基类。正如前面提到的,在Java平台上AnyRef实际就是类java.lang.Object的别名。因此Java里写的类和Scala里写的都继承自AnyRef。

存在AnyRef别名代替使用java.lang.Object名称的理由是,Scala被设计成可以同时工作在Java和.Net平台。在.NET平台上,AnyRef是System.Object的别名。

可以认为java.lang.Object是Java平台上实现AnyRef的方式。因此,尽管你可以在Java平台上的Scala程序里交换使用Object和AnyRef,推荐的风格是在任何地方都只使用AnyRef。

Scala类与Java类不同在于它们还继承自一个名为ScalaObject的特别的记号特质。理念是ScalaObject包含了Scala编译器定义和实现的方法,目的是让Scala程序的执行更有效。到现在为止,Scala对象包含了单个方法,名为$tag,用于内部以提速模式匹配。

原始类型是如何实现的

Scala以与Java同样的方式存储整数:把它当作32位的字。这对在JVM上的效率以及与Java库的互操作性方面来说都很重要。标准的操作如加法或乘法都被实现为原始操作。然而,当整数需要被当作(Java)对象看待的时候,Scala使用了“备份”类java.lang.Integer。如在整数上调用toString方法或者把整数赋值给Any类型的变量时,就会这么做。

所有这些听上去都近似Java5里的自动装箱并且它们的确很像。不过有一个关键差异,Scala里的装箱比Java里的更少看见。尝试下面的Java代码:

  // This is Java
  boolean isEqual(int x, int y) {
    return x == y;
  }
  System.out.println(isEqual(421, 421));

当然会得到true。现在,把isEqual的参数类型变为java.lang.Integer(或Object,结果都一样):

  // This is Java
  boolean isEqual(Integer x, Integer y) {
    return x == y;
  }
  System.out.println(isEqual(421, 421));

却得到了false!原因是数421被装箱了两次,因此参数x和y是两个不同的对象。

因为在引用类型上==表示引用相等,而Integer是引用类型,所以结果是false。这是展示了Java不是纯面向对象语言的一个方面。我们能清楚观察到原始类型和引用类型之间的差别。

现在在Scala里尝试同样的实验:

  scala> def isEqual(x: Int, y: Int) = x == y
  isEqual: (Int,Int)Boolean

  scala> isEqual(421, 421)
  res10: Boolean = true

  scala> def isEqual(x: Any, y: Any) = x == y
  isEqual: (Any,Any)Boolean

  scala> isEqual(421, 421)
  res11: Boolean = true

实际上Scala里的相等操作==被设计为透明的参考类型代表的东西。对值类型来说,就是自然的(数学或布尔)相等。对于引用类型,==被视为继承自Object的equals方法的别名。这个方法被初始地定义为引用相等,但被许多子类重载实现它们种族的相等概念。这也意味着Scala里你永远也不会落入Java知名的关于字串比较的陷阱。Scala里,字串比较以其应有的方式工作:

  scala> val x = "abcd".substring(2)
  x: java.lang.String = cd

  scala> val y = "abcd".substring(2)
  y: java.lang.String = cd

  scala> x == y
  res12: Boolean = true

Java里,x与y的比较结果将是false。程序员在这种情况应该用equals,不过它容易被忘记。

然而,有些情况你需要使用引用相等代替用户定义的相等。

例如,某些时候效率是首要因素,你想要把某些类哈希合并(hash cons)然后通过引用相等比较它们的实例(类实例的哈希合并是指把创建的所有实例缓存在弱集合中。然后,一旦需要类的新实例,首先检查缓存。如果缓存中已经有一个元素等于你打算创建的,你可以重用存在的实例。这样安排的结果是,任何以equals()判断相等的两个实例同样在引用相等上判断一致。)。

为这种情况,类AnyRef定义了附加的eq方法,它不能被重载并且实现为引用相等(也就是说,它表现得就像Java里对于引用类型的==那样)。同样也有一个eq的反义词,被称为ne。例如:

  scala> val x = new String("abc")
  x: java.lang.String = abc

  scala> val y = new String("abc")
  y: java.lang.String = abc

  scala> x == y
  res13: Boolean = true

  scala> x eq y
  res14: Boolean = false

  scala> x ne y
  res15: Boolean = true

底层类型

层级的底部你看到了两个类scala.Null和Scala.Nothing。它们是用统一的方式处理某些Scala的面向对象类型系统的“边界情况”的特殊类型。

类Null是null类型的引用;它是每个引用类(就是说,每个继承自AnyRef的类)的子类。Null不兼容值类型。你不可,比方说,把null值赋给整数变量:

  scala> val i: Int = null
  <console>:4: error: type mismatch;
   found   : Null(null)
   required: Int

类型Nothing在Scala的类层级的最底端;它是任何其它类型的子类型。然而,根本没有这个类型的任何值。要一个没有值的类型有什么意思呢?在控制结构的try-catch中讨论过,Nothing的一个用处是它标明了不正常的终止。例如Scala的标准库中的Predef对象有一个error方法,如下定义:

  def error(message: String): Nothing =
    throw new RuntimeException(message)

error的返回类型是Nothing,告诉用户方法不是正常返回的(代之以抛出了异常)。因为Nothing是任何其它类型的子类,你可以非常灵活的使用像error这样的方法。例如:


  def divide(x: Int, y: Int): Int = 
    if (y != 0) x / y 
    else error("can't divide by zero")

if状态分支,x / y,类型为Int,而else分支,调用了error,类型为Nothing。因为Nothing是Int的子类型,整个状态语句的类型是Int,正如需要的那样。

特质(Traits)

trait是Scala里代码复用的基础单元。特质封装了方法和字段的定义,并可以通过混入到类中重用它们。

概念

定义, trait NoPoint(x: Int, y: Int) // Does not compile:

  trait Philosophical {
    def philosophize() {
      println("I consume memory, therefore I am!")
    }
  }

没有声明超类,因此和类一样,有个缺省的超类AnyRef。一旦特质被定义了,就可以使用extends或with关键字,把它混入到类中:

  class Frog extends Philosophical {
    override def toString = "green"
  }

类Frog是AnyRef(Philosophical的超类)的子类并混入了Philosophical。从特质继承的方法可以像从超类继承的方法那样使用:

  scala> val frog = new Frog
  frog: Frog = green

  scala> frog.philosophize()
  I consume memory, therefore I am!

特质同样也是类型。以下是把Philosophical用作类型的例子:

  scala> val phil: Philosophical = frog
  phil: Philosophical = green

  scala> phil.philosophize()
  I consume memory, therefore I am!

phil的类型是Philosophical,一个特质。因此,变量phil可以被初始化为任何混入了Philosophical特质的类的对象。

如果想把特质混入到显式扩展超类的类里,可以用extends指明待扩展的超类,用with混入特质:

  class Animal

  class Frog extends Animal with Philosophical {
    override def toString = "green"
  }

如果想混入多个特质,都加在with子句里就可以了:

  class Animal
  trait HasLegs

  class Frog extends Animal with Philosophical with HasLegs {
    override def toString = "green"
  }

目前为止你看到的例子中,类Frog都继承了Philosophical的philosophize实现。或者,Frog也可以重载philosophize方法。语法与重载超类中定义的方法一样。举例如下:

  class Animal

  class Frog extends Animal with Philosophical {
    override def toString = "green"
    override def philosophize() {
      println("It ain't easy being "+ toString +"!")
    }
  }

因为Frog的这个新定义仍然混入了特质Philosophize,你仍然可以把它当作这种类型的变量使用。但是由于Frog重载了Philosophical的philosophize实现,当你调用它的时候,你会得到新的回应:

  scala> val phrog: Philosophical = new Frog
  phrog: Philosophical = green

  scala> phrog.philosophize()
  It ain't easy being green!

这时你或许推导出以下哲理:特质就像是带有具体方法的Java接口,不过其实它能做的更多。特质可以,比方说,声明字段和维持状态值。实际上,你可以用特质定义做任何用类定义做的事,并且语法也是一样的,除了两点。第一点,特质不能有任何“类”参数,也就是说,传递给类的主构造器的参数。换句话说,尽管你可以定义如下的类:

  class Point(x: Int, y: Int)

但是下面定义特质的尝试将遭到失败:

  trait NoPoint(x: Int, y: Int) // Does not compile

类和特质的另一个差别在于不论在类的哪个角落,super调用都是静态绑定的,在特质中,它们是动态绑定的。如果你在类中写下“super.toString”,你很明确哪个方法实现将被调用。然而如果你在特质中写了同样的东西,在你定义特质的时候super调用的方法实现尚未被定义。调用的实现将在每一次特质被混入到具体类的时候才被决定。这种处理super的有趣的行为是使得特质能以可堆叠的改变(stackable modifications)方式工作的关键。

比较瘦接口与胖接口

瘦接口与胖接口的对阵体现了面向对象设计中常会面临的在实现者与接口用户之间的权衡。胖接口有更多的方法,对于调用者来说更便捷。客户可以捡一个完全符合他们功能需要的方法。另一方面瘦接口有较少的方法,对于实现者来说更简单。然而调用瘦接口的客户因此要写更多的代码。由于没有更多可选的方法调用,他们或许不得不选一个不太完美匹配他们所需的方法并为了使用它写一些额外的代码。

Java的接口常常是过瘦而非过胖。例如,从Java 1.4开始引入的CharSequence接口,是对于字串类型的类来说通用的瘦接口,它持有一个字符序列。下面是把它看作Scala特质的定义:

  trait CharSequence {
    def charAt(index: Int): Char
    def length: Int
    def subSequence(start: Int, end: Int): CharSequence
    def toString(): String
  }

尽管类String成打的方法中的大多数都可以用在任何CharSequence上,Java的CharSequence接口定义仅提供了4个方法。如果CharSequence代以包含全部String接口,那它将为CharSequence的实现者压上沉重的负担。任何实现Java里的CharSequence接口的程序员将不得不定义一大堆方法。因为Scala特质可以包含具体方法,这使得创建胖接口大为便捷。

在特质中添加具体方法使得胖瘦对阵的权衡大大倾向于胖接口。不像在Java里那样,在Scala中添加具体方法是一次性的劳动。你只要在特质中实现方法一次,而不再需要在每个混入特质的方法中重新实现它。因此,与没有特质的语言相比,Scala里的胖接口没什么工作要做。

要使用特质丰满接口,只要简单地定义一个具有少量抽象方法的特质——特质接口的瘦部分——和潜在的大量具体方法,所有的都实现在抽象方法之上。然后你就可以把丰满了的特质混入到类中,实现接口的瘦部分,并最终获得具有全部胖接口内容的类。

样例:长方形对象

为了使这些长方形对象便于使用,如果库能够提供诸如width,height,left,right,topLeft,等等方法。

没有特质的代码的话,首先会有一些基本的集合类如Point和Rectangle:

  class Point(val x: Int, val y: Int)

  class Rectangle(val topLeft: Point, val bottomRight: Point) {
    def left = topLeft.x
    def right = bottomRight.x
    def width = right - left
    // and many more geometric methods...
  }

这个Rectangle类在它的主构造器中带两个点,分别是左上角和右下角的坐标。然后它通过对这两个点执行简单的计算实现了许多便捷方法诸如left,right,和width。 图库应该有的另一个类是2-D图像工具:

  abstract class Component {
    def topLeft: Point
    def bottomRight: Point

    def left = topLeft.x
    def right = bottomRight.x
    def width = right - left
    // and many more geometric methods...
  }

注意left,right,和width在两个类中的定义是一模一样。除了少许的变动外,他们将在任何其他的长方形对象的类中保持一致。

这种重复可以使用特质消除。这个特质应该具有两个抽象方法:一个返回对象的左上角坐标,另一个返回右下角的坐标。然后他就可以应用到所有其他的几何查询的具体实现中

  trait Rectangular {
    def topLeft: Point
    def bottomRight: Point

    def left = topLeft.x
    def right = bottomRight.x
    def width = right - left
    // and many more geometric methods...
  }

类Component可以混入这个特质并获得Rectangular提供的所有的几何方法:

  abstract class Component extends Rectangular {
    // other methods...
  }

可以创建Rectangle对象并对它调用如width或left的几何方法:


  scala> val rect = new Rectangle(new Point(1, 1),
       | new Point(10, 10))
  rect: Rectangle = Rectangle@3536fd

  scala> rect.left
  res2: Int = 1

  scala> rect.right
  res3: Int = 10

  scala> rect.width
  res4: Int = 9

实现Ordered特质来排序

Scala专门提供了Ordered特质解决排序。以前面用过的分数类型来说,原来已经有了四个比较大小的方法:

  class Rational(n: Int, d: Int) {
    // ...
    def < (that: Rational) =
      this.numer * that.denom > that.numer * this.denom
    def > (that: Rational) = that < this
    def <= (that: Rational) = (this < that) || (this == that)
    def >= (that: Rational) = (this > that) || (this == that)
  }

现在通过Ordered特质的compare方法来完成一个比大小的功能:

  class Rational(n: Int, d: Int) extends Ordered[Rational] {
    // ...
    def compare(that: Rational) =
      (this.numer * that.denom) - (that.numer * this.denom)
  }

上面定义compare方法来比较两个对象,类Rational现在具有了所有4种比较方法:

  scala> val half = new Rational(1, 2)
  half: Rational = 1/2

  scala> val third = new Rational(1, 3)
  third: Rational = 1/3

  scala> half < third
  res5: Boolean = false

  scala> half > third
  res6: Boolean = true

考虑下面的抽象简化了四个比较操作符的实现:

  trait Ordered[T] {
    def compare(that: T): Int

    def <(that: T): Boolean = (this compare that) < 0
    def >(that: T): Boolean = (this compare that) > 0
    def <=(that: T): Boolean = (this compare that) <= 0
    def >=(that: T): Boolean = (this compare that) >= 0
  }

特质的继承

可以把多个特质加在一个类上,按声明的顺序,它们操作的顺序就像栈一样。

IntQueue有一个put方法把整数添加到队列中,和一个get方法移除并返回它们:

  abstract class IntQueue {
    def get(): Int
    def put(x: Int)
  }

BasicIntQueue类根据上面的特质实现了一个队列:

  import scala.collection.mutable.ArrayBuffer

  class BasicIntQueue extends IntQueue {
    private val buf = new ArrayBuffer[Int]
    def get() = buf.remove(0)
    def put(x: Int) { buf += x }
  }

运行一个队列的效果:

  scala> val queue = new BasicIntQueue
  queue: BasicIntQueue = BasicIntQueue@24655f

  scala> queue.put(10)

  scala> queue.put(20)

  scala> queue.get()
  res9: Int = 10

  scala> queue.get()
  res10: Int = 20

Doubling特质把整数放入队列的时候对它加倍。

  trait Doubling extends IntQueue {
    abstract override def put(x: Int) { super.put(2 * x) }
  }

他定义了超类IntQueue这个定义意味着特质只能混入到扩展了IntQueue的类中。声明为抽象的方法中有一个super调用。这种调用对于普通的类来说是非法的,因为他们在执行时将必然失败。然而对于特质来说,这样的调用实际能够成功。因为特质里的super调用是动态绑定的,特质Doubling的super调用将直到被混入在另一个特质或类之后,有了具体的方法定义时才工作。

所以要加上abstract override,说明要继承父特质(只有特质,类里是不能这样写的)才能实际产生作用。

现在一行代码都不用写,下面只有一个extends和一个with就把类定义好了:

  scala> class MyQueue extends BasicIntQueue with Doubling
  defined class MyQueue

  scala> val queue = new MyQueue
  queue: MyQueue = MyQueue@91f017

  scala> queue.put(10)

  scala> queue.get()
  res12: Int = 20

这个队列即能入队出队,而且数字还是加倍的。

还可以更加简化到类名都不用写,直接`new BasicIntQueue with Doubling`这个父类名加上特质就可以把对象拿到了:

  scala> val queue = new BasicIntQueue with Doubling
  queue: BasicIntQueue with Doubling = $anon$1@5fa12d

  scala> queue.put(10)

  scala> queue.get()
  res14: Int = 20

特质的叠加

有两个特质,一个在入队时把值加1;另一个过滤掉负数:

  trait Incrementing extends IntQueue {
    abstract override def put(x: Int) { super.put(x + 1) }
  }

  trait Filtering extends IntQueue {
    abstract override def put(x: Int) {
      if (x >= 0) super.put(x)
    }
  }

队列能够即过滤负数又对每个进队列的数字增量:

  scala> val queue = (new BasicIntQueue
       | with Incrementing with Filtering)
  queue: BasicIntQueue with Incrementing with Filtering...

  scala> queue.put(-1); queue.put(0); queue.put(1)

  scala> queue.get()
  res15: Int = 1

  scala> queue.get()
  res16: Int = 2

粗略地说,越靠近右侧的特质越先起作用。。如果那个方法调用了super,它调用其左侧特质的方法,以此类推。

前面的例子里,Filtering的put首先被调用,因此它移除了开始的负整数。Incrementing的put第二个被调用,因此它对剩下的整数增量。

如果你逆转特质的次序,那么整数首先会加1,然后如果仍然是负的才会被抛弃:

  scala> val queue = (new BasicIntQueue
       | with Filtering with Incrementing)
  queue: BasicIntQueue with Filtering with Incrementing...

  scala> queue.put(-1); queue.put(0); queue.put(1)

  scala> queue.get()
  res17: Int = 0

  scala> queue.get()
  res18: Int = 1

  scala> queue.get()
  res19: Int = 2

为什么不是多重继承

特质是一种继承多个类似于类的结构的方式,但是它与许多语言中的多重继承有很重要的差别。其中的一个尤为重要:super的解释。对于多重继承来说,super调用导致的方法调用可以在调用发生的地方明确决定。而对于特质来说,方法调用是由类和被混入到类的特质的线性化(linearization)所决定的。这种差别让前一节所描述的改动的堆叠成为可能。

在关注线性化之前,请花一点儿时间考虑一下在传统的多重继承语言中如何堆叠改动。假想有下列的代码,但是这次解释为多重继承而不是特质混入:

  val q = new BasicIntQueue with Incrementing with Doubling
  q.put(42) // which put would be called?

第一个问题是,哪个put方法会在这个调用中被引用?或许规则会决定最后一个超类胜出,本例中的Doubling将被调用。Doubling将加倍它的参数并调用super.put,大概就是这样。增量操作将不会发生!同样,如果规则决定第一个超类胜出,那么结果队列将增量整数但不会加倍它们。因此怎么排序都不会有效。

或许你会满足于允许程序员显式地指定在他们说super的时候他们想要的到底是哪个超类方法。比方说,假设下列Scala类似代码,super似乎被显式地指定为Incrementing和Doubling调用:

  trait MyQueue extends BasicIntQueue
      with Incrementing with Doubling {

    def put(x: Int) {
      Incrementing.super.put(x) // (Not real Scala)
      Doubling.super.put(x)
    }
  }

这种方式将带给我们新的问题。这种尝试的繁冗几乎不算是问题。实际会发生的是基类的put方法将被调用两次——一次带了增量的值另一次带了加倍的值,但是没有一次是带了增量加倍的值。

显然使用多重继承对这个问题来说没有好的方案。你不得不返回到你的设计并分别提炼出代码。相反,Scala里的特质方案很直接。你只要简单地混入Incrementing和Doubling,Scala对super的特别照顾让它迎刃而解。这与传统的多重继承相比必然有不同的地方,但这是什么呢?

就像在前面暗示的,答案就是线性化。当你使用new实例化一个类的时候,Scala把这个类和所有它继承的类还有它的特质以线性:linear的次序放在一起。然后,当你在其中的一个类中调用super,被调用的方法就是链子的下一节。除了最后一个调用super之外的方法,其净结果就是可堆叠的行为。

线性化的精确次序由语言的式样书描述。虽然有一点儿复杂,但你需要知道的主旨就是,在任何的线性化中,某个类总是被线性化在所有其超类和混入特质之前。因此,当你写了一个调用super的方法时,这个方法必将改变超类和混入特质的行为,没有其它路可走。

Scala的线性化的主要属性可以用下面的例子演示:假设你有一个类Cat,继承自超类Animal以及两个特质Furry和FourLegged。FourLegged又扩展了另一个特质HasLegs:

  class Animal
  trait Furry extends Animal
  trait HasLegs extends Animal
  trait FourLegged extends HasLegs
  class Cat extends Animal with Furry with FourLegged

继承关系

类型的线性化看起来是这样的:

Animal : Animal, AnyRef, Any
Furry : Furry, Animal, AnyRef, Any
FourLegged : FourLegged, HasLegs, Animal, AnyRef, Any
HasLegs : HasLegs, Animal, AnyRef, Any
Cat : Cat, FourLegged, HasLegs, Furry, Animal, AnyRef, Any

因为Animal没有显式扩展超类或混入任何超特质,因此它缺省地扩展了AnyRef,并随之扩展了Any。

Animal : Animal, AnyRef, Any

第二部分是第一个混入,特质Furry的线性化,但是所有已经在Animal的线性化之中的类现在被排除在外,因此Cat的线性化中每个类仅出现1次。结果是:

Furry : Furry, Animal, AnyRef, Any

它之前是FourLegged的线性化,任何已被复制到线性化中的超类及第一个混入再次被排除在外:

FourLegged : FourLegged, HasLegs, Animal, AnyRef, Any

最后,Cat线性化的第一个类是Cat自己:

当这些类和特质中的任何一个通过super调用了方法,那么被调用的实现将是它线性化的右侧的第一个实现。

Cat : Cat, FourLegged, HasLegs, Furry, Animal, AnyRef, Any

什么情况下要用特质

没有固定的规律,但是包含了几条可供考虑的导则。

如果行为不会被重用,那么就把它做成具体类。具体类没有可重用的行为。

如果要在多个不相关的类中重用,就做成特质。只有特质可以混入到不同的类层级中。

如果你希望从Java代码中继承它,就使用抽象类。因为特质和它的代码没有近似的Java模拟,在Java类里继承特质是很笨拙的。而继承Scala的类和继承Java的类完全一样。除了一个例外,只含有抽象成员的Scala特质将直接翻译成Java接口,因此即使你想用Java代码继承,也可以随心地定义这样的特质。要了解让Java和Scala一起工作的更多信息请看后面其他的章节。

如果你计划以编译后的方式发布它,并且你希望外部组织能够写一些继承自它的类,你应更倾向于使用抽象类。原因是当特质获得或失去成员,所有继承自它的类就算没有改变也都要被重新编译。如果外边客户仅需要调用行为,而不是继承自它,那么使用特质没有问题。

如果效率非常重要,倾向于类。大多数Java运行时都能让类成员的虚方法调用快于接口方法调用。特质被编译成接口,因此会付出微小的性能代价。然而,仅当你知道那个存疑的特质构成了性能瓶颈,并且有证据说明使用类代替能确实解决问题,才做这样的选择。

包和引用

通过把package子句放在文件顶端的方式把整个文件内容放进包里:

  package bobsrockets.navigation
  class Navigator 

另一种方式很像C#的命名空间。在package子句之后用大括号包起来一段要放到包里去的定义。除此之外,这种语法还能把同一个文件内的不同部分放在不同的包里。

  package bobsrockets {
    package navigation {

      // In package bobsrockets.navigation
      class Navigator

      package tests {

        // In package bobsrockets.navigation.tests
        class NavigatorSuite
      }
    }
  } 

类似于Java的语法实际上只是括号嵌入风格的语法糖。原理是:如果除了签入另一个包之外对包不作任何事,你可以下面的方式省去一个缩进:

  package bobsrockets.navigation {

    // In package bobsrockets.navigation
    class Navigator

    package tests {

      // In package bobsrockets.navigation.tests
      class NavigatorSuite
    }
  } 

Java包尽管是分级的,却不是嵌套的。在Java里,在你命名一个包的时候,你必须从包层级的根开始。

Scala为了简化,采用包风格类似于是相对路径:

  package bobsrockets {
    package navigation {
      class Navigator
    }
    package launch {
      class Booster {
        // No need to say bobsrockets.navigation.Navigator
        val nav = new navigation.Navigator
      }
    }
  } 

而且内部区域的包可以隐匿被定义在外部区域的同名包。还提供了_root_表示顶层包:

  // In file launch.scala
  package launch {
    class Booster3
  }

  // In file bobsrockets.scala
  package bobsrockets {
    package navigation {
      package launch {
        class Booster1
      }
      class MissionControl {
        val booster1 = new launch.Booster1
        val booster2 = new bobsrockets.launch.Booster2
        val booster3 = new _root_.launch.Booster3
      }
    }
    package launch {
      class Booster2
    }
  } 

上面的代码中,为了访问Booster3,Scala提供了所有用户可创建的包之外的名为_root_的包。换句话就是,任何你写的顶层包都被当作是_root_包的成员。

因此,_root_.launch让你能访问顶层的launch包,_root_.launch.Booster3指向的就是最外面的booster类。

引用包

Scala里用import子句来引用包和其成员:

  package bobsdelights

  abstract class Fruit(
    val name: String,
    val color: String
  )

  object Fruits {
    object Apple extends Fruit("apple", "red")
    object Orange extends Fruit("orange", "orange")
    object Pear extends Fruit("pear", "yellowish")
    val menu = List(Apple, Orange, Pear)
  } 

不止是包,其他成员也可以用import来引用:

  // easy access to Fruit
  import bobsdelights.Fruit

  // easy access to all members of bobsdelights
  import bobsdelights._

  // easy access to all members of Fruits
  import bobsdelights.Fruits._ 

差别是Scala的按需引用写作尾下划线_而不是星号*(毕竟*是合法的Scala标识符!)。上面的第三个引用子句与Java的静态类字段引用一致。

cala引用可以出现在任何地方,而不是仅仅在编译单元的开始处。同样,它们可以指向任意值。

  def showFruit(fruit: Fruit) {
    import fruit._
    println(name +"s are "+ color)
  } 

方法showFruit引用了它的参数,Fruit类型的fruit,的所有成员。之后的println语句就可以直接使用namecolor了。这两个索引等价于fruit.namefruit.color

Scala的引用很灵活的另一个方面是它们可以引用包自身,而不只是非包成员。这只有你把内嵌包想象成包含在外围包之内才是自然的。

例如,下面的代码里包java.util.regex被引用。这使得regex可以用作简单名。要访问java.util.regex包的Pattern单例对象,你可以只是写成regex.Pattern

  import java.util.regex

  class AStarB {
    // Accesses java.util.regex.Pattern
    val pat = regex.Pattern.compile("a*b")
  } 

Scala的引用同样可以重命名或隐藏成员。可以用跟在引用的成员对象之后的包含在括号里的引用选择子句(import selector clause)做到:

  • 重命名子句的格式是<原始名> => <新名>`。
  • <原始名> => _格式的子句从被引用的名字中排除了<原始名>。
// 只引用了对象Fruits的Apple和Orange成员。 
import Fruits.{Apple, Orange} 

// Apple对象重命名为McIntosh。
import Fruits.{Apple => McIntosh, Orange}

// 以SDate的名字引用了SQL的日期类,因此你可以在同时引用普通的Java日期类Date。
import java.sql.{Date => SDate}

// 以名称S引用了java.sql包,这样你就可以写成S.Date。
import java.{sql => S}

// 引用了对象Fruits的所有成员。这与import Fruits._同义。
import Fruits.{_}

// 从Fruits对象引用所有成员,不过重命名Apple为McIntosh。
import Fuites.{Apple => McIntosh, _}

// 引用了除Pear之外的所有Fruits成员。
import Fuits.{Pear => _, _}

把某样东西重命名为_就是表示把它隐藏掉。这对避免出现混淆的局面有所帮助。比方说你有两个包,Fruits和Notebooks,它们都定义了类Apple。如果你想只是得到名为Apple的笔记本而不是水果,就需要引用所有的Notebooks和除了Apple之外所有的水果:

  import Notebooks._
  import Fruits.{Apple => _, _} 

总而言之,引用选择可以包括下列模式:

  • 简单名x。把x包含进引用名集。
  • 重命名子句x => y。让名为x的成员以名称y出现。
  • 隐藏子句x => _。把x排除在引用名集之外。
  • 全包括‘_’。引用除了前面子句提到的之外的全体成员。如果存在全包括,那么必须是引用选择的最后一个。

本节最初展示的比较简单的引用子句可以被视为带有选择子句的简写。例如,import p._等价于import p.{_}并且import p.n等价于import p.{n}

隐式引用

Scala隐式地添加了一些引用到每个程序中。本质上,就好象下列的三个引用子句已经被加载了:

  import java.lang._ // everything in the java.lang package
  import scala._     // everything in the scala package
  import Predef._    // everything in the Predef object 

java.lang包囊括了标准Java类。它永远被隐式包含在Scala的JVM实现中。.NET实现将代以引用system包,它是java.lang的.NET模拟。

scala包含有标准的Scala库,包括许多通用的类和对象。因为scala被隐式引用,你可以直接用List而不是scala.List

Predef对象包含了许多Scala程序中常用到的类型,方法和隐式转换的定义。比如,因为Predef是隐式引用,你可以直接写assert而不是Predef.assert

上面的这三个引用子句与其它的稍有不同,靠后的引用将遮盖靠前的。

例如,StringBuilder类被定义在scala包里以及Java版本1.5以后的java.lang包中都 有。因为scala引用遮盖了java.lang引用,所以StringBuilder简单名将被看作是scala.StringBuilder,而不是java.lang.StringBuilder

访问修饰符

Scala与Java对访问修饰符的对待方式有一些重要的差异。

private

标记为private的成员仅在包含了成员定义的类或对象内部可见。但不同于Java,Scala里这个规则同样应用到了内部类上。这种方式更一致。

  class Outer {
    class Inner {
      private def f() { println("f") }
      class InnerMost {
        f() // OK
      }
    }
    (new Inner).f() // error: f is not accessible
  } 

Scala里,(new Inner).f()访问非法,因为f在Inner中被声明为private而访问不在类Inner之内。相反,类InnerMost里访问f没有问题,因为这个访问包含在Inner类之内。

这点与Java不同,Java会允许这两种访问因为它允许外部类访问其内部类的私有成员。

protected

Java中protected可以被同一个包里的类访问。Scala里,只在定义了成员的类的子类中可以被访问。

  package p {
    class Super {
      protected def f() { println("f") }
    }
    class Sub extends Super {
      f()
    }
    class Other {
      (new Super).f()  // error: f is not accessible
    }
  } 

public

没有任何标记就是公开的,可以在任何地方被访问。

保护的范围

Scala里的访问修饰符可以通过使用修饰词增加。格式为private[X]protected[X]的修饰符表示针对“X”的私有或保护,这里“X”指代某些外围的包,类或单例对象。

这样允许定义一些在你项目的若干子包中可见但对于项目外部的客户却始终不可见的东西。同样的技巧在Java里是不可能的:

 package bobsrockets {
   package navigation {
     private[bobsrockets] class Navigator { 
       protected[navigation] def useStarChart() {}
       class LegOfJourney {
         private[Navigator] val distance = 100
       }
       private[this] var speed = 200
     }
   }
   package launch {
     import navigation._
     object Vehicle { 
       private[launch] val guide = new Navigator
     }
   }
 } 

类Navigator被标记为private[bobsrockets]。这就是说这个类对包含在bobsrockets包的所有的类和对象可见。特别是对象Vehicle里对Navigator的访问被允许,因为Vehicle包含在包launch中,而launch包在bobsrockets中。另一方面,包bobsrockets包之外的所有代码都不能访问类Navigator。

private修饰词同样可以直接是外围包。对象Vehicle类的guide对象的访问修饰符是这样的例子。这种访问修饰符等价于Java的包私有访问。

没有修饰符 公开访问
Private[bobsrockets] 在外部包中访问
Private[navigation] 与Java的包可见度相同
Private[Navigator] 与Java的private相同
Private[LegOfJourney] 与Scala的private相同
Private[this] 仅在同一个对象中可以访问

所有的修饰词也可以用在protected上,与private意思相同。也就是说,类C的protected[X]修饰符允许C的所有子类和外围的包、类、或对象X访问被标记的定义。例如,useStarChart方法在Navigator所有子类以及包含在navigation包里的所有代码能够被访问。这与Java的protected意思完全一致。

Private的修饰词还能指向外围类或对象。例如LegOfJourney里的distance变量被标记为private[Navigator],因此它在类Navigator的任何地方都可见。这与Java里的内部类的私有成员具有同样的访问能力。private[C]里的C如果是最外层类,那么private的意思和Java一致。

Scala还具有一种比private更严格的访问修饰符。被private[this]标记的定义仅能在包含了定义的同一个对象中被访问。这种定义被称为对象私有:object-private。例如,代码13.11中,类Navigator的的speed定义就是对象私有的。这就是说所有的访问必须不仅是在Navigator类里,而且还要是同一个Navigator实例发生的。

因此在Navigator内访问“speed”和this.speed是合法的。然而以下的访问,将不被允许,即使它发生在Navigator类之中:

  val other = new Navigator
  other.speed // this line would not compile 

把成员标记为private[this]是一个让它不能被同一个类中其它对象访问的保障。这在做文档时比较有用。有时它也能让写出更通用的变体注释(参见以后会讲到的参数类型化章节的变体注释)。

伴生对象的可见度性

Java里,静态成员和实例成员属于同一个类,因此访问修饰符可以统一地应用在他们之上。在Scala里没有静态成员,代之以可以拥有包含成员的仅存在一个的伴生对象。

  class Rocket {
    import Rocket.fuel
    private def canGoHomeAgain = fuel > 20
  }

  object Rocket {
    private def fuel = 10
    def chooseStrategy(rocket: Rocket) {
      if (rocket.canGoHomeAgain)
        goHome()
      else
        pickAStar()
    }
    def goHome() {}
    def pickAStar() {}
  }
 

Scala的访问规则给予了伴生对象和类一些特权。类把它所有的访问权限共享给半生对象,反过来也是如此。特别的是,对象可以访问所有它的伴生类的私有成员,就好象类也可以访问所有伴生对象的私有成员一样。

举个例子,上面的Rocket类可以访问方法fuel,它在Rocket对象中被声明为私有。类似地,Rocket对象也可以访问Rocket类里面的私有方法canGetHome。

有一个例外,说到protected static成员时,Scala和Java的相似性被打破了。Java类C的保护静态成员可以被C的所有子类访问。相反,伴生对象的protected成员没有意义,因为单例对象没有任何子类。

断言和测试

断言

assert是预定义的方法,assert(condition)会在条件不成立时抛出AssertionError

另一个版本assert(condition, explanation),指定了抛出含有指定explantion作为说明的assertionErrorexplation的类型为Any,所以任何对象都可以作为参数。assert方法会对传入的参数调用toString的结果作为放在AssertionError中的文字描述。

在前面的“组合与继承”这一章中的above方法加上道检查,只有宽度一样的元素才可以上下连接在一起:

  def above(that: Element): Element = {
    val this1 = this widen that.width
    val that1 = that widen this.width
    elem(this1.contents ++ that1.contents)
  }

assert方法所在的Predef包里还有一个ensuring方法可以直接对表达式的返回结果进行测试而不用先把表达式的结果先存放到变量中:

  def widen(w: Int): Element =
    if (w <= width) this
    else {
      val left = elem(' ', (w - width) / 2, height)
      var right = elem(' ', w - width - left.width, height)
      left beside this beside right
    } ensuring (w <= _.width)

ensuring接收一个返回类型为Boolean的函数作为参数。注意这里ensuring是作用在if-else表达式的Element类结果上的,而不是方法widen的返回值上。相当于是:

  def widen(w: Int): Element = {
    if (w <= width) this
    else {
      val left = elem(' ', (w - width) / 2, height)
      var right = elem(' ', w - width - left.width, height)
      left beside this beside right
    } ensuring (w <= _.width)
  }

还有一点要明白的是,这里的语法看起来像是对Element类对象调用ensuring方法(把xxx.ensuring(...)中的点换成了空格)。但是实际上Emement对象是没有这个成员方法的,而是被隐式转换成了Ensuring对象。由于存在隐式转换,所以ensuring`可以作用在任何类型上。

测试

受益于和Java之间无缝操作。Java的测试库,像JUnit、TestNG,可以直接用于scala的单元测试。ScalaTest是专为scala设计的一个测试框架。ScalaTest提供了一些比较方便的功能。

注意ScalaTest除了自己的版本号外,对应不同的Scala版本还有不同的对应版本。拿错了是用不了的。

ScalaTest

ScalaTest提供了函数式的FunSuite(Function Suite)。test方法为测试方法,说明字符串的内容会显示在输出信息中:

import org.scalatest.FunSuite
import scala.collection.mutable.Stack
 
class ExampleSuite extends FunSuite {
 
  test("pop is invoked on a non-empty stack") {
    val stack = new Stack[Int]
    stack.push(1)
    stack.push(2)
    val oldSize = stack.size
    val result = stack.pop()
    assert(result === 2)
    assert(stack.size === oldSize - 1)
  }
 
  test("pop is invoked on an empty stack") {
    val emptyStack = new Stack[Int]
    intercept[NoSuchElementException] {
      emptyStack.pop()
    }
    assert(emptyStack.isEmpty)
  }
}
\-(morgan:%) >>> scalac -cp scalatest_2.9.0-1.9.1.jar ExampleSuite.scala                      
\-(morgan:%) >>> scala -cp .:scalatest_2.9.0-1.9.1.jar org.scalatest.run ExampleSuite 
Run starting. Expected test count is: 2
ExampleSuite:
- pop is invoked on a non-empty stack
- pop is invoked on an empty stack
Run completed in 158 milliseconds.
Total number of tests run: 2
Suites: completed 1, aborted 0
Tests: succeeded 2, failed 0, ignored 0, pending 0
All tests passed.

Junit 4

如果喜欢JUnit4的风格,那么可以写下面这样的单元测试。

import org.scalatest.junit.AssertionsForJUnit
import scala.collection.mutable.ListBuffer
import org.junit.Assert._
import org.junit.Test
import org.junit.Before

class ExampleSuite extends AssertionsForJUnit {

  var sb: StringBuilder = _
  var lb: ListBuffer[String] = _

  @Before 
	def initialize() {
    sb = new StringBuilder("ScalaTest is ")
    lb = new ListBuffer[String]
  }

  @Test 
	def verifyEasy() { // Uses JUnit-style assertions
    sb.append("easy!")
    assertEquals("ScalaTest is easy!", sb.toString)
    assertTrue(lb.isEmpty)
    lb += "sweet"
    try {
      "verbose".charAt(-1)
      fail()
    }
    catch {
      case e: StringIndexOutOfBoundsException => // Expected
    }
  }

  @Test 
	def verifyFun() { // Uses ScalaTest assertions
    sb.append("fun!")
    assert(sb.toString === "ScalaTest is fun!")
    assert(lb.isEmpty)
    lb += "sweeter"
    intercept[StringIndexOutOfBoundsException] {
      "concise".charAt(-1)
    }
  }
}

编译运行:

\-(morgan:%) >>> scalac -cp scalatest_2.9.0-1.9.1.jar:junit-4.8.2.jar ExampleSuite.scala                      
\-(morgan:%) >>> scala -cp .:scalatest_2.9.0-1.9.1.jar:junit-4.8.2.jar org.junit.runner.JUnitCore ExampleSuite
JUnit version 4.8.2
..
Time: 0.026

OK (2 tests)

混合ScalaTest与JUnit

org.scalatest.junit.JUnitSuite已经混入了特质AssertionsForJUnit,可以同时被用于JUnit与ScalaTest方法:

import org.scalatest.junit.JUnitSuite
import scala.collection.mutable.ListBuffer
import org.junit.Assert._
import org.junit.Test
import org.junit.Before

class ExampleSuite extends JUnitSuite {

  var sb: StringBuilder = _
  var lb: ListBuffer[String] = _

  @Before
	def initialize() {
    sb = new StringBuilder("ScalaTest is ")
    lb = new ListBuffer[String]
  }

  @Test
	def verifyEasy() { // Uses JUnit-style assertions
    sb.append("easy!")
    assertEquals("ScalaTest is easy!", sb.toString)
    assertTrue(lb.isEmpty)
    lb += "sweet"
    try {
      "verbose".charAt(-1)
      fail()
    }
    catch {
      case e: StringIndexOutOfBoundsException => // Expected
    }
  }

  @Test
	def verifyFun() { // Uses ScalaTest assertions
    sb.append("fun!")
    assert(sb.toString === "ScalaTest is fun!")
    assert(lb.isEmpty)
    lb += "sweeter"
    intercept[StringIndexOutOfBoundsException] {
      "concise".charAt(-1)
    }
  }
}

JUnit调用:

\-(morgan:%) >>> scala -cp .:scalatest_2.9.0-1.9.1.jar:junit-4.8.2.jar org.junit.runner.JUnitCore ExampleSuite
JUnit version 4.8.2
..
Time: 0.026

OK (2 tests)

ScalaTest调用:

\-(morgan:%) >>> scala -cp .:scalatest_2.9.0-1.9.1.jar:junit-4.8.2.jar org.scalatest.run ExampleSuite
Run starting. Expected test count is: 2
ExampleSuite:
- verifyEasy
- verifyFun
Run completed in 226 milliseconds.
Total number of tests run: 2
Suites: completed 1, aborted 0
Tests: succeeded 2, failed 0, ignored 0, pending 0
All tests passed.

ScalaTest还提供了函数式的单元测试。

// 用函数式的方式来写单元测试
// IDE目前对ScalaTest的支持不是特别好
// 加上RunWith就可以用JUnit的方式来运行了
@RunWith(classOf[JUnitRunner])
class ElementSuite3 extends FunSuite {
    test("elem result should have passed width") {
        val ele = elem('x', 2, 3)
        assert(ele.width == 2)
    }
}

单元测试对提高软件质量很有好处。唯一的不足就是只针对程序员。其它人员要看懂还是比较有困难。ScalaTest提供了BDD(Behavior Driven Development行为驱动开发)测试方式。下面的这段测试代码在运行时就会打印出可读的解释。

class ElementSpec extends FlatSpec with ShouldMatchers {
    "A UniformElement" should
        "have a width equal to the passed value" in {
        val ele = elem('x', 2, 3)
        ele.width should be(2)
    }
    it should
        "have a height equal to the passed value" in {
        val ele = elem('x', 2, 3)
        ele.height should be(3)
    }
    it should
        "throw an IAE if passed a negative width" in {
        evaluating {
            elem('x', -2, 3)
        } should produce[IllegalArgumentException]
    }
}

上面的代码会打印出下面这样的提示。

A UniformElement 
- should have a width equal to the passed value
- should have a height equal to the passed value
- should throw an IAE if passed a negative width

我们写单元测试时会测试一些边界值。然后再选一些典型的值。如果这些选值有库来做,不但可以减少单元测试的工作量,而且可以将边界值选取更合理。下面是如何将ScalaChecker和ScalaTest联合起来使用的一个例子。

class ElementSpecChecker extends FlatSpec with ShouldMatchers with Checkers{
    "A UniformElement" should
        "have a width equal to the passed value" in {
        // 这可以用数学化的方式来读
        // 对每个整数w
        // 当w>0时
        // 都有后面的等式成立
        check((w: Int) => w > 0 ==> (elem('x', w, 3).width == w))
    }
}
Category: scala | Tags: Scala
11
25
2012
0

halo3攻略

光晕3攻略(不了解光晕3剧情的进)转贴

 
作者:get (UT攻略组)
译 名: 光环3
原 名: Halo 3
机 种: X360
类 型: FPS
开发商: Bungie Software
发行商: microsoft game studios
发售日: 2007年9月25日
售 价: 60美元
在闪烁的夜空中,一颗陨石划破了这华丽的天际,向着地面陨落.
"我没说过是他们让我选的吗?我可以自行挑选超级战士"
"你应该知道我一直都在观察,看着你逐渐成为我们理想的战士"
"很其他人一样,你很强壮,敏捷又勇敢,是天生的领导者"
"但是我发现,你有一点是他们所没有的"
"知道是什么吗?"
"运气"
此时流星一分为二,另一半高速冲向了地面并发生爆炸
"不是吗?
地球士兵迅速来到了"流星"坠落的地方,其实那不应该是流星,而是我们的士官长.士官长的雷神锤MARK-IV盔甲吸收了这次降落的大部分冲击,
也因为这样,当地球士兵士官长正处于一个盔甲锁死状态,医疗兵迅速解除了锁死状态,但士官长好象因为强大的冲击而昏迷了,正当上士拆除了
士官长头盔后的精片,准备让吊车把士官长运走的时候,士官长突然醒来了.士官长拿回了精片,突然间,眼前闪出CONTANA的话 "不要随便对女
生作承诺,如果你没把握做到".......这时候要根据士兵的话上下移动镜头来解除盔甲锁定,正当一切都调整完毕,准备上战场之际,一个透明的身
影从士兵后边出现,士官长出于本能第一时间拿着枪冲上去指着"星盟战士"的下额 "士官长,等等,神风列士不是敌人!"JOHNSON及时冲过来制
止了士官长 "拜托!麻烦已经够多的了,你们俩别再自伤残杀!" 士官长松开了手,神风列士活动了一下刚才被枪指着的下额 "真那么简单就好" "快
走吧,鬼面兽嗅到我们了"说完就带领着士官长和地球士兵们离开森林,踏上了最后一战的旅程
任务目标
1.前往河边准备撤退
2.找出JOHNSON坠毁的地点
3.在JOHNSON被俘虏前找到他
4.拯救JOHNSON和他的小队
5.撑着点!飞船快到了
伸展筋骨
刚开始熟悉一下操作,然后跟着神风列士走,来到一瀑布前会看见一星盟的飞船从头顶掠过,之后的敌人就是一批一批的出现了,注意中间星盟飞船下人
的时候会有重机枪扫射,消灭完路上的敌人快要到达河边的时候再次闪出CONTANA的影像 "你会为了任务而牺牲我吗? 你会眼睁睁看着我死吗?"
气死人,太可恶了
在河边看到星盟的人在和地球军交火,尽管已经迅速到达,但敌人强大的火力还是将地球军的飞船击落,这里敌人众多,从左边的小路迅速饶过去把敌人
消灭是个不错的选择,从左边进去掩护物也特别多,来到飞船的坠毁的地点,发现里边的人全部都消失了,看来还是来晚了一步,在坠落的飞船那拿一支猎
击枪后,通过坠机附近的桥来到一个被星盟占领的基地,在远处看见被俘虏的JOHNSON,先用刚才得到的猎击枪把基地前方的敌人扫一下,注意还要留10颗
子弹,然后到右边的通道那会第一次看见一种拿锤的敌人,锤子的攻击非常大,一下就可以把士官长打死,一看见他就用猎击枪把他从远距离打死,来到关
押俘虏的地方,里边有充足的武器,先补充完再对着中央的装置按RB键就能把JOHNSON放出来,这时候敌人的后续部队也会到达,找个掩护物躲好,等后续
部队过来后迅速消灭掉,我们地球的飞船也到达了,迅速奔上飞船,本关结束.
来到基地,士兵们看到士官长都十分振奋,与指挥官接触并了解了部分现在的战况并得知真相先知正在寻找某个叫方舟的地方,只要他找到方舟,就可以
在那点燃所有的HALO环带,正当众人与HOOD统帅了解下一步的行动时,基地突然大范围停电,而屏幕上出现了一个星盟的面孔 "你们以为你们能阻止即将
到来的毁灭?别妄想了,一切都会彻底化为灰烬,连你们的恶魔同盟,也无法阻挠我们的朝圣之旅,真是痛快啊!上天要你们毁灭,而我只是替天行道罢了"
这时候基地的电源已经恢复,那丑陋的面孔也消失在大屏幕中,看来基地已经暴露了,别攻击只是时间的问题,指挥官当机立断指示众人撤退......
  • 1楼
  • 2009-04-05 03:56
 
任务目标
1.确认周围防御工事的安全
2.消灭停机棚的敌人
3.返回作战中心
4.拯救阵地内的陆战队员
5.经由升降台撤退
6.回到控制中心启动炸弹
7.脱逃-利用停机棚的升降机
认清自我...
由于基地是旧时建筑,所以门都需要通过手动来打开,这里的门都需要长按RB键来开启,消灭路上的敌人,来到尽头却因为缺少口令而无法进入,沿路从中
间的门进入到停机棚,迅速找到机枪的位置,然后用机枪清扫敌人,当星盟飞船飞进来的时候应该尽快把船头的机关炮打掉,然后再慢慢消灭落下的敌人,
把敌人全部清理掉后,拔走机器枪往基地内部走回控制中心,控制中心内有一大堆冰蜂,慢慢小心的攻击,不难应付
买一送一
在指挥中心看到一个大炸弹,指挥官想送给星盟一个惊喜的饯别礼,跟着JOHNSON旁边的士兵来到后方的门,现在的任务是救出陆战队员,并带他们离开,
一开门就遇见到量的"猩猩士兵"刚才的机枪在这就大发神威了,用他迅速消灭掉前边的四个星盟敌人,然后用手雷快速把拿锤的星盟敌人炸死,如果没弹
药的话,右边有个武器箱,内有充足的武器.来到内部再次看见CONTANA的影像 "你被征召入....."来到下边和神风列士汇合后进入前方的门,在里边会看
见生还的士兵,这里如果爬到上层来打,会方便很多,而且敌人一般不会对上层做太大的攻击,来到升降台并启动它,来到停机棚遇到一批会飞的星盟敌人
,清理完等友军都上机后接到新的任务,回去控制中心启动炸弹.
最后走的人别忘了关灯
从左边的地区回去,进门前看到CONTANA的影像 "你的责任就是保护地球及其殖民地",一开门迅速把敌人杀死并拔走电浆炮,路上有很多电浆炮,建议一
路杀完就拔走,来到控制中心,优先把最远的那个星盟士兵除掉,因为他的武器太猛了,然后把附近的敌人全部消灭掉,启动炸弹后迅速下楼.在路上再次
出现CONTANA的影像 "这条路上将会有重重困难等着你" "你将成为我们最顶尖的战士"回到刚开始的停机棚,从右边的升降机下去,刚一进去,外边就传
来的洪洪烈火,在升降机下降的时候再次出现CONTANA的影像 "这里将是你的家" 升降机飞速下落,在底部还发生爆炸...难道真的如CONTANA说的那
句 "最后你也将会在这里安息"吗........
任务目标
1.带领陆战队员离开基地
2.快到沃域去
漫漫长路
醒来后发现一批陆战队员,这时候收到指挥官的指示,要求你找到交通工具离开基地,在后边的仓库能找到几辆装甲车,上车后带领着陆战队员一直杀出
去,本关可以说,大部分时间都是在车上度过的,中途的敌人只要操纵灵活,基本上没太多的威胁,穿过隧道后路因为桥的折断车无法再驶进,沿路进去火
力支缓,出现敌人WRAITH装甲车的时候应该冲过去近身把车破坏掉,路上会发现一种新的星盟交通工具,而且该处于敌人多数也是使用该种交通工具的,
建议把他们引诱在悬崖边,让他们自己冲下去,而中途的WRAITH最好抢夺,因为后边一段路用他来清理会比较轻松.进入最后部分的敌人比较多,而且都是
皮特别厚的敌人,注意找好掩体,优先干掉炮台,剩下的敌人用步枪远距离慢慢射即可.
任务目标
1.消灭第一架亡魂号
2.消灭所有残存的亡魂号
3.消灭圣甲虫号
4.破坏防空炮
鬼域
先到门的右边按RB键把门打开,上二楼左边可以换武器,从二楼右边过去消灭完敌人后给下边的机车开门,在里边消灭了一批敌人后就可以上车前进了,
出去后上空会有一辆敌人的运兵船,下边有两辆WRAITH装甲车,先上我们自己的机车用机枪把地面的敌人都清扫一便,然后迅速近身走到WRAITH装甲车前
按RB上去按B键直接把车打掉.进入下一区域,从门一进去直接往左走饶进去,由于下边的我军会吸引一部分火力,趁机会从侧面把敌人都消灭,进入下一
区域开门的时候会碰到大批冰蜂,用二楼的机枪快速消灭,需要补充弹药的话附近有武器箱,进入门的时候再次看到CONTANA残存的影像 "我...早就不信
  • 2楼
  • 2009-04-05 03:56
 
鬼神了",下一区域会有两量敌人运兵船,运兵船会放出几辆敌人的BANSHEE,趁运兵船降低的时候冲过去对其内部一顿猛烈攻击,就能把船打落.
审判
下面将会面对圣甲虫号,刚开始开着车在它身边饶一会儿,等一听到场上出先DU~DU的声音的时候立刻从它屁屁的入口冲上去直接攻击其核心,这是比较
方便的一种方法,但如果你觉得这种方法没挑战性,可以从墙边按B把火箭发射器拔走,然后冲过去在甲虫下边对着它的脚关节不断狂轰,这种方法相对刺
激许多!收拾完这大家伙后收到命令,要进入另一区域破坏防空炮,以便HOOD统帅发动进攻,进去基地后再次看到CONTANA残存的影像 "我是你的盾,也是
你的剑",在这里补充下弹药,并拔下机枪,一进仓库迅速解决眼前的几个敌人,前进到出口的时候会遇到两个装甲十分厚的敌人,因为行动比较缓慢,建议
扔几个手雷加几枪解决他们吧.解决他们后拐两个弯就见到敌人的防空炮,防空炮发射后会露出核心,给他的核心不断开火就能把防空炮打坏了.
终于破坏了敌人的防空炮,HOOD统帅也得以发动攻击,一轮火力全开的炮轰后,本以为能够把敌人的遗址消灭,但突然间遗址开始下降,并对天空射出了一
条光,所有的人都被光的冲击波而吹飞,而这时候,士官长的脑海中也响起了CONTANA的声音 "这就是世界毁灭的方式....."
当士官长醒来的时候发现天空出现了一个环状的光圈,"究竟真相先知就了些什么",正当众人都准备思考这个问题的时候,看来真相先知是不会给时间让
士官长考虑,一辆星盟的飞船冲破了空间,从士官长头上划过
"那是什么?鬼面兽吗?"
"恐怕更糟...."
任务目标
1.找到坠毁的虫族船舰
2.在船舰中找到CONTANA的下落
伴我回家
刚才飞下来的飞船里边载着的东西果然如士官长所说,是更恐怖的虫族,现在唯一消灭虫族的方法是要找到坠落飞船的核心并把它引爆,让其毁灭所有的
虫族.回到刚才沿路经过的基地,这里已经变成了一片火海,与友军碰面后遇到的是大量虫族,由于虫族数量众多,所以建议采取步步为营的方法,千万不
能着急,把仓库里的虫族都消灭光后,发现进来的大门已经被关上了,从二楼上去你会发现一个大洞,跳进去后会看到CONTANA残存的影像 "我无法告诉你
每件事..." 途中会遇到一个士兵,从他口中得知其他所有的士兵都已经被感染了,冲出去后突然,空中出现了一个巨大的武舰.....
鬼影号战舰
鬼影号的人放下了一些星盟士兵来帮士官长消灭虫族,顺路杀进去,由于现在的路都比较窄,很容易就会被几个虫族包围,所以建议用远程武器把能看见
的敌人都清理一下再前进,在中途会得知CONTANA就在刚才那艘船上,路上注意有一种新的敌人,内久比较高,建议用双枪迅速打掉,部分敌人会在高处射
击,应当优先解决,在到坠毁的战舰,从中间的大洞跳下去.
无穷的罪恶机器
一直往下走,在最下边的装置内看到CONTANA,准备将他回收的时候一个SPARK从空中飞下来,他说如果现在不把CONTANA修理好,CONTANA就会毁灭,士官长
无奈的看着CONTANA,只好把她暂时交给SPARK,而自己也跟随着SPARK进入了星盟的战舰.
SPARK迅速修理好了CONTANA,CONTANA一出现就说:"博爱之城,也是先知的圣域,正朝着地球前进,还带着庞大的虫族!我没办法把所有事都告诉你,这里不
安全,尸脑兽知道我在系统中" "但是,他并不晓得传送门通往何处,在另一边,有一个方法可以阻止虫族,并且不会启动其他HALO环带"话音未落,CONTANA
的影像就痛苦倒地,她支撑着身体,告诉士官长 "快点,士官长,快找到[方舟],时间不多了."说完,影像就消失了.星盟的人听完这段话决定穿过那道门,
去对付鬼面兽和真相先知,HOOD统帅则率领剩余部队,守着人类最后的防线,抵抗到最后一秒,而士官长则去寻找CONTANA所说的解决方法,并把她带回
来......
任务目标
1.消灭敌人的防空火力
2.带领友军穿过先行者之墙
3.找出制图机
  • 3楼
  • 2009-04-05 03:56
 
4.到楼下去准备撤退
00坐标
坐着鬼影号战舰飞过了那闪门,迎接众人的是数量多三倍鬼面兽,降落后找到敌人的一个阵地,从高处利用地形优势,用猎击枪迅速把他们的大家伙都消
灭掉,后边的基地也是如此,经过山洞后会遇到一队坠落的友军,他们的运兵船被敌人包围,把他们消灭掉后会有两小队的敌人坐着车过来,把他们消灭掉
后开车,冲到一个基地后把里边的敌人都消灭掉,发现那里的门现在都无法打开,之后会提示要你给指挥官找登陆区,沿路回去会有一个地点有四辆
WRAITH,把他们全部摧毁后,指挥官就能降落了.飞船壮观地降落后会放出坦克,坐上坦克,由于坦克的攻击力巨大,敌人也变得不堪一击,回到刚才过不去
的墙,SPARK会去打开那扇门,然后跟着它前进,给友军打开通道后一打开门,迎接众人的是在地球已经面对过一次的圣甲虫号,把地面的敌人全部消灭后
就能对于这大怪物了,由于有了上次的经验,而且这次还有坦克,对于它简直易如反掌,一开始只要冲进甲虫底部,对着他的腿猛轰,等他DU~DU的时候立刻
移动到他外边对准他的核心轰炸,圣甲虫号再次倒下了.到旁边的塔顶与神风列士汇合.
强者不需要看地图
进入门后敌人全部都睡着了,用近身技把他们全部消灭掉,里边的敌人都不多,而且还有大量星盟的武器,在里边找到终端机,一直往下走就能找到制图机
.从制图机那得知,士官长现在正处于银河系9光年外的一个名叫[方舟]的地方,而[方舟]核心附近的防护栏已经被启动,士官长必须穿过防护栏,要不然
外来者会毁了它.这时候两辆鬼面战机突然飞来,众人只后撤退了,注意下边开始的敌人会隐形,注意看雷达才能够发现他们,在撤退点有大量敌人,建议
到左边拿上星盟的武器再去攻击.
任务目标
1.关闭第一具防护塔
2.关闭最后一具防护塔
3.破坏要塞
4.阻止真相先知启动环带
5.虫族来了,快逃命吧!
一下飞机就有大批敌人在等待,先把枪换成BR55,把眼前的几个敌人先消灭掉,然后从右边来到瀑布边,用M6把远处的对空装甲车毁掉,等友军把机车都放
下后驾驶车,进去消灭掉敌人后进入防护塔内部,到二楼坐升降机上去,把里边的敌人全部消灭掉后到尽头把塔关闭.正当士官长关掉第一个防护塔的时
候,神风列士也成功把第二个防护塔关闭掉,之后得到命令要与神风列士汇合再前往第三座塔,从沿路回去,开车回到海滩会看见友军已经开着黄蜂号来
了,驾驶上黄蜂号,从右边的海岸继续前进.
凡事都得自己来...
开着飞机就以为着空战,把空中的敌人都清理干净后,注意还得把地面的部队都顺便消灭干净才能降落,下去后和神风列士汇合,里边有两个装甲鬼面兽,
还赠送一大堆黄蜂,消灭掉下边的敌人就迅速上楼,关闭最后一具防护塔后,天空中突然出现了CONTANA刚才所说的博爱之城,一堆虫族从天而降,并坠毁
在士逛长面前,料理好面前一大堆虫族后下去,到外边和友军汇合后开车前往并破坏要塞.
旅途终点
一路驾驶坦克把路上的敌人全部杀光后,来到一处宽阔的地面处会出现两辆圣甲虫号,此时立刻换上黄蜂号,冲过去饶着圣甲虫号攻击,等他不能移动的
时候迅速打掉他的保护盖并破坏它的核心,当然,如果你想强行登陆的话也是没问题的,下地面与神风列士汇合,SPARK会开启通往要塞的桥,进入内部后
出现CONTANA的残存影像 "我只是有问必答,为了一时的安全,我竟为整个宇宙,带来毁灭...." 进入里边后听到真相先知的话,看来他已经准备启动环带
了,并且从影像中看到被俘虏的JOHNSON,看来速度要加快了.正当真相先知想逼JOHNSON干掉什么的时候,指挥官突然驾驶飞船冲了进来,但面对着人数众
多的敌人,指挥官也无能为力,正当指挥官和JOHNSON准备自杀以不让环带启动的时候指挥官却先被真相先知偷袭而身亡,而真相先知更利用JOHNSON之手
开启了环带!而这时候士官长也赶来了,突然出现的虫族对士官长说现在只有他能终结真相先知所做的事,
  • 4楼
  • 2009-04-05 03:56

 
真相大白
一路杀过去,虽然敌人的数量众多,但这时候虫族会帮忙开路,所以也不需要太紧张,杀到尽头.
JOHNSON抱着已经死去的指挥官,对着士官长说:"你要阻止环带的启动,救助其他人,神风列士捉住了已经被感染并奄奄一息的真相先知,他临死前还
在说着大条救世之道,士官长走上了启动开关,把环带系统关闭了,而神风列士也给真相先知送上了致命一击,了结了他的生命.这时候巨大的触手突然冲出,
士官长拉着神风列士坐飞机准备逃出,但最终也被触手打落下来.
沿路返回,在尽头的洞口跳下,下落后士官长好象看到了CONTANA的影子,并在尽头打开了开关,出现在他们俩面前的,是士官长曾经摧毁过的环
带替代品,看来要彻底消灭虫族,只有启动眼前这个熟悉的物体.......
任务目标
1,找到CONTANA
2.摧毁反应炉
3.逃出博爱之城
泛滥猖獗
一路从洞穴中下去,尽头会看到一个触手一般的洞口,进去后会有幻影 "无敌之子你为何要来,父债子还啊",再往下走,来到一个平台处,这里有大量虫族
,由于数量实在太多,只能建议你多用手雷等武器开路,如果没子弹的话,换上剑用近身也是个不错的选择.途中虫族首领会不断骚扰,来到一个大厅里的
时候往大厅右上方向走,通关两条长长的楼梯终于来到一个大厅,中间安放着一个球装仪器,按B键把他打烂.
迎接士官长的,是那个与他并肩战斗的声音:"你找到我了."士官长看着仪器内虚弱的CONTANA "我..已经严重受损,你太迟了."士官长爬下了身子,对着CONTANA
说:"你知道的,当我做了承诺...""你...不会食言,很高兴看到你,士官长." "真幸运,那东西你还留着吗?" CONTANA站了起来:"第一个HALO环带的启动索引
器."CONTANA伸出了手,她的手上出现了那个索引器 "我随身携带这小纪念品以防万一."她看了看身边的环境,问士官长:"有脱逃计划吗?" "我本来是想杀条血路,
轰动一点."士官长从头后取出了那块精片,CONTANA回到了精片里,她再次回到了士官长身边了. "冷静点,别冲动,不仅有你,我也在这,记得吗?"
超越地狱之火
回到中央的大厅启动中间的装置,然后把四边的反应炉都破坏掉,这时候CONTANA会提示爆炸给你炸出了一条出路,顺着箭头下去,路上回去在之前进入一
个大厅里会有一条路,走过去的时候会有一只虫族冲出来,灭了后沿着他冲出来的地方走,一直走下去,在入口的地方会遇到神风列士,原来是神风列士来
救士官长来了,跑到悬崖边用之前坠毁的飞机逃出了这个鬼地方....
任务目标
1.找出控制室
2.杀了引导者,启动HALO[环带]
3.快上护卫舰
士官长和神风列士驾驶这飞机,缓缓得向着空中那巨大的环带设施前进,飞船坠落在环带里,这个环带看起来应该是还没完成的,现在的目标是找到传送
门...沿着雪地一路前进,来到控制塔下边的时候会遇到大批虫族,右边有路可以进入塔内,路上会遇到大量虫族的攻击,来到第三层,需要你把一波虫族
都消灭掉才能开门,由于三层的场景比较空,敌人会从两边进攻,一定要注意走位和射击的精确程度!实在不行就用神风列士做诱惑在他后边攻击吧.三人
进入内部后看见环带,JOHNSON和CONTANA去毁灭环带,而士官长和神风列士则在外边看风,这时候SPARK会出现,并攻击想启动装置的JOHNSON,并攻击了前
来救缓的士官长,SPARK露出了他本来的面目,看来不消灭掉着东西就不能启动环带了.SPARK不怎么厉害,只要注意他的光束攻击(注意跑位)和冲击波(没
伤害,但会把你推下桥),基本上可以做到无伤,捡起JOHNSON留下来的激光炮三/四下来就能干掉他了.JOHNSON轰轰烈烈地死去了,死前把CONTANA的精片
交回给了士官长 "千万不能失去她..."
世界末日
环带终于启动了,原路出去,从门的右边的冰路上走过去,进入右边的门,顺着箭头一直走,来到外边会看见JOHNSON留下的机车,最后就是一段疯狂的飞车
之旅,主要还是留意陨石会撞毁地下的甲板,不要掉下去就行.
"如果我们失败"
"我们会成功的"
"JOHN,很荣幸跟你合作"
回到地球后,人类已经开始着手重建,正当HOOD统帅等人满怀希望的等着士官长的回归,但走出仓门的,却只有神风列士....
神风列士来到阵亡列士碑上,上边有JOHNSON等一众士兵的照片,HOOD统帅走到神风列士面前 "我仍旧无法原谅你们的过错,不要忘记这场战争是你们挑
起的.不过.....我很感谢你陪伴他战斗到最后一刻" 在阵亡列士碑上,有三个数字,这三个数字纪念着那个为了地球的和平和献身的战士.....

但,他...真的阵亡了吗?
在冷冻仓内,士官长对着CONTANA说
"有事就叫醒我"
-END-
Category: 未分类 | Tags:
11
25
2012
0

Halo2攻略

标题: 《光晕2》的剧情攻略
作者:   2006年07月08日 23:27:37  【 】 【发表评论/查看评论

宇宙,深邃浩瀚的宇宙,充满着凶险和神秘;


  宇宙,星光熠熠的宇宙,遍布着历险与传奇。


  Chapter 0 - The Fall of Reach 瑞奇星的沦陷


   经过几个世纪,超光速已实现。在联合国星际指挥部(United Nations Space Command,UNSC)全力推动移民计划下,数以百万计的人开始移民外太空。移民计划的重心则在一个星球上瑞奇星(Reach)。那是一个中继站,为 平民建造宇宙飞船,也为UNSC制造星际战舰。由于十分接近地球,瑞奇星也成为了科技及军事行动的中心。


  公元2525年, 一个殖民地哈维斯星(Harvest)的联系忽然中断,UNSC派往调查的一个战斗群几乎全军覆灭,只有一艘严重受损的战舰返回瑞奇星,回报是一艘外星战 舰毫不费力地摧毁了他们。这是人类与被称为“圣约人(Covenant)”的外星军团的第一次接触。圣约人是一个宗教狂热的外星种族。圣约长老们宣称,人 类冒犯了他们的神,因此要发动一场针对人类的血腥圣战。


  人类的军事力量完全无法与圣约人抗衡,殖民地一个接一个被摧毁,数百万人类成了圣约人的炮灰。


  在数个殖民星球被摧毁之后,UNSC的军事将领柯尔(Cole)上将订定了一个柯尔协议:“……任何船只均不得以任何理由将圣约人引到地球去;如果必须撤退,船只必须避免经过任何邻近地球的星系,即使缺乏适当的导航亦然;如果在可能被俘虏的情形下,船只必须自毁!”


   这时,在瑞奇星上,UNSC开始了一个秘密军事计划建立一支被称为斯巴坦-II(Spartan)的超级战士部队,他们都身穿一种由秘密材料开发的,可 抵挡圣约人的武器的MJORNIR超强装甲。但由于这支部队的人数太少,并不足以扭转战局。之后,斯巴坦-II战士们被召回瑞奇星进行整编,以执行一个重 要任务登上一艘圣约战舰,设法找出圣约人的根据地。不幸地,在任务开始的两天前,圣约人对瑞奇星发动大规模进攻,斯巴坦-II战士几乎全军覆没。公元 2552年,圣约人摧毁了瑞奇星。


  圣约人离地球只有一步之遥了。此时,一艘人类战舰秋之舰(Pillar of Autumn)载着最后一个斯巴坦-II战士,向外层空间执行了一次超光速空间跳跃,希望能把圣约人引离地球,愈远愈好……


Chapter 1 - Pillar of Autumn 袭击秋之舰 (Normal难度,下同)


   耀眼的银河映衬着巨大的瑞梭特星(Threshold),旁边,有一个淡蓝色的环状物。秋之舰(Pillar of Autumn)正在缓缓接近这个星体。奇斯舰长(Captain Keyes)正在听取这首飞船的人工智能柯塔娜(Cortana)的汇报。很不幸,他们并没有摆脱圣约人的追击。

舰长下令全员回覆到一级战备状态,同时下令放出最后一个斯巴坦-II计划的战士马斯达队长(Master Chief)。他是人类对抗圣约人的最终武器了。


  在冷冻室,技术员打开队长的冷冻舱。马斯达队长走出,并在技术员引导下完成各种测试,但是突然圣约人闯入了秋之舰!舰长下令马斯达队长立即赶往舰桥。于是队长连枪都来不及拿就跟着技术员赶往舰桥,可惜那个可怜的技术员一出门就……Bomm……


  来到舰桥,舰长自知寡不敌众,于是下令按照柯尔协议把秋之舰坠毁到那个环形物上面。舰长抽出柯塔娜的晶片,让队长放在头盔里,叫柯塔娜连接上自己的神经信号。这一切都是为了不让圣约人得到人类的资料和地球的位置。最后,舰长把手枪给了队长,命令大家找救生艇逃生。


  队长有了枪在手,马上跑出去杀杀杀……最后终于找到一艘救生艇,和余下的人类部队逃出秋之舰。


救生艇掠过千疮百孔秋之舰,飞向那个巨大的环形物体,在神秘的环形世界中,等待他们的将会是什么呢?



  Chapter 2 - Halo 环形世界


  救生艇急速冲进环形星体的大气层,坠毁在地面。马斯达队长在柯塔娜的呼叫下慢慢苏醒,救生艇里面的战士都遇难了,武器药包散落一地。


   这个环形世界竟然和地球没什么区别,有山,有海,有植被。远处,一段巨大的环臂底部与广阔的海洋接壤,上部一直延伸上天空,在头顶的天幕环绕一圈,在大 地的另一边尽头再次与大地接壤。右边天空是巨大的瑞梭特星(Threshold),左边是它的卫星比西斯星(Basis),这个环形世界就处于两个星体之 间。这是宇宙的创造物,还是人工的艺术品呢?


  柯塔娜报告附近有圣约人的飞船(Dropship)接近,看来圣约人也一早登陆了这个环。柯塔娜又建议去寻找别的生还者。


  清除敌人后,柯塔娜收到一艘人类鹈鹕战舰(Pelican)艾高(Echo)419号的呼叫,驾驶员是弗哈玛(Foe Hammer)。柯塔娜提议她和队长去寻找生还士兵,由艾高419协助运送。艾高419放下一辆疣猪战车,有两位战士自愿和队长一起去寻找生还者。


   开车穿过一个山洞。山里面的建筑都是人工制造的,难道这里有过什么大工程?柯塔娜侵入了圣约人的战网,监视他们的行动,同时不断尝试联络奇斯舰长的神经 信号。另外,柯塔娜发现更多的人员逃离了秋之舰登陆到这个环上,于是叫队长继续寻找生还者,以便集结力量,组成统一战线对付圣约人。


  然后就是队长一边开车兜风一边拯救生还士兵……


  柯塔娜得知圣约人已经找到秋之舰的坠毁地,并且俘虏了所有生还者,包括奇斯舰长。他们把舰长囚禁在一艘名为“真相和解号(Truth and Reconsilliation)”的圣约巡洋舰里面。于是,大家乘艾高419去营救舰长。



Chapter 3 - Truth and Reconsilliation 真相和解号


   环形星体在挂满星斗的漆黑天幕中闪着淡蓝色的光。山顶上停泊着一艘圣约人巡洋舰“真相和解号”。柯塔娜探测到舰长的神经信号是从里面发出来的。艾高 419咆哮着飞过悬崖,队长他们这次为了营救奇斯舰长而深入虎穴,将会和圣约人展开一场殊死的战斗。艾高419就会再次派援军下来。利用红色的重力升降粒 子束登上“真相和解号”。


  巡洋舰里面一片沉寂,突然周围四个红色的门杀出大批敌人……沿路不断杀戮,记得保住几个队友啊不然你会打得很辛苦,不过在停泊港那里,艾高419还会再派人来的,最后找到舰长的囚室。


   队长拉起奇斯舰长,在秋之舰坠毁到这个环形星体上到现在才第一次重逢,大家非常激动。舰长告诉队长,圣约人称这个环形的星体为“光晕”,光晕和他们的宗 教有某种联系。圣约人相信光晕其实是一种可怕的武器,还说谁控制了光晕,谁就控制了宇宙。柯塔娜又说她从圣约人的战网中得知他们在找光晕的控制室。于是, 舰长分配了一个新任务给马斯达队长和柯塔娜,阻止圣约人找到光晕的控制室,不然,如果光晕真是一件威力无比的武器,圣约人就可以用它来毁灭地球了……


  不过,现在迫在眉睫的是得先离开圣约人的虎穴。柯塔娜尝试叫艾高419来接应,但是弗哈玛说路上遇到圣约人的截击,不能脱身。于是舰长说只要能找到一艘圣约人的Dropship就可以离开了,柯塔娜记得停泊港三楼有开关,于是大家火速沿路返回停泊港。




Chapter 4 - Silent Cartographer 无声地图绘制员


   浩瀚的海面在瑞梭特星照耀下闪着粼粼波光,两艘鹈鹕舰艾高419和布莱沃22,正掠过海面加速飞向一个小岛。与舰长分道扬镳后,柯塔娜捕获圣约人战网的 资讯,他们正在寻找一个“无声地图绘制员”,他们相信这个“绘制员”就在海岛的地底。最重要的是,这个“地图绘制员”可以指示光晕的控制室的位置。


  海水轻拂着岸边,大群圣约敌兵开始向鹈鹕舰疯狂进攻。扫清这个海滩的敌人后,艾高419会放一辆疣猪战车下来,载上两个战友就可以继续扫荡这个岛了。战士们都聚集在这片海滩,如果车上的战友有什么意外,还可以回来这里再搭两个的。


  在路上所见的景观,明显说明这个岛不是天然的,而是一个非常庞大的海上人造设施,只不过外表铺了一层岩石和植被而已。但是究竟是谁建造的呢?


  关闭岛上的防御装置后,队长单人匹马下去找“绘图员”。一直杀到最底层,见到“无声绘图员”原来是光晕的全息图。


  队长启动控制面版,光晕全息图立即分成很多个小块,看起来光晕是由很多不同的区域组成的。柯塔娜分析到光晕控制室的坐标,她想联络奇斯舰长,但是联络不上,只好叫艾高419在地面接应。

现在必须赶紧沿原路回到地面,抢在圣约人之前找到控制室。


  根据“绘图员”的坐标,柯塔娜建议艾高419从地底的隧道飞行。她发现圣约人对整个光晕进行了彻底的地震扫描,得知光晕内部是一个布满深坑隧道的结构。圣约人难道发现了光晕内部有什么秘密?

Chapter 5 - Assault On the Control Room 控制室之战



   这里的天气和前面截然不同,竟下起了大雪。按道理来说,光晕的纬度跨度很窄,时间间隔又那么短,不可能出现差别那么大的天气的。但柯塔娜怀疑这里的天气 才是一个有大气的星体上真实的天气状况,而前面那些是因为有某种调节器的作用,才会出现热带或温带气候。这里下大雪,唯一可能的解释就是天气调节器出了故 障。但是,为什么光晕上会有气候调节器呢?

人类的疣猪战车,天蝎坦克(Scorpion);圣约人的鬼魂摩托(Ghost),女妖战机(Banshee)大混战,实在是爽……不过也不要光自己爽,要有团结精神,顾顾你的队友的小命……

历 尽艰辛,终于进入控制室内。一个巨大的光晕全息图环绕着房间,中间是小型的瑞梭特星、光晕、比西斯星的天体系统全息图。队长把柯塔娜的晶片插入终端,柯塔 娜的图像立即显现。她发现,光晕并非只是一件武器那么简单,而是有更深刻含义的东西,是一群被称为“先人(Forerunner)”的种族创造了它……但 在她能领悟解释光晕到底是什么之前,她突然发现了一个危机在光晕的内部,埋藏着一种非常可怕的东西,圣约人已经发现了这种东西,他们感到非常恐慌。更糟的 是,奇斯舰长要去寻找的所谓武器隐藏地,事实上就是这种东西的埋藏地!

时间无多,柯塔娜已经来不及解释,她叫队长马上出发,一定要赶在那种可怕的东西到达不可控制的局面之前,阻止舰长的行动!马斯达队长冲出控制室,没有带走柯塔娜。




Chapter 6 - 343 Guilty Spark 343妖火



  在一片浓雾弥漫的沼泽中,一群圣约人似乎正在逃避某种看不见的敌人。艾高419缓缓降落,弗哈玛说奇斯舰长的飞船信号最后是从这里发出已经是12小时前的事了。她会等队长找到舰长后再来接应。


  队长断断续续的收到一段来自一艘维克托933鹈鹕舰的求救信号,内容大概是在沼泽的某个建筑物里,舰长和战士们受到袭击,而对方……并不是圣约人?!

进入那个建筑物。转来转去头都晕了,奇怪的是除了一些尸体,什么敌人都没有?不久又见到一个小房间中有个士兵发了狂的向你开火,他似乎见过某种可怕的东西,受了巨大的刺激。不用管他(也可以把准星对准他……免得他痛苦),最后来到一个门前。

队长发现这个门装了一个电子锁,打开后里面竟跌出一具士兵的尸体!房间里没有其它人,只是听到一些很奇怪的吱吱声。队长捡起一个头盔,拆下里面的录像记录,放进自己的头盔里播放。


  录像记录了舰长那一支部队突入这栋建筑的经过,他们最后也是来到这个房间,但他们却遇到了另一种敌人的袭击那是一种像章鱼一样的绿色小生物,数量极多。它们不但袭击人类,而且还袭击了圣约人……在录像结束时,只知道录像的士兵也不幸牺牲,而舰长生死未卜……

情况非常不妙,得马上逃离这个鬼地方。但这时房间内的门纷纷被破开,那种绿色小生物铺天盖地涌进来……后来竟然还有人形的怪物闯进来……估计是被那种小生物侵蚀了的人类或圣约人的寄生体,太可怕了!即使是刀枪不入的英雄有时候也得要……逃命啊!!


  从另一台升降机回到地面。艾高419说附近有座高塔,必须在那里才能接应。于是队长和士兵们一起杀向高塔。但是,队长在高塔附近却被传送了上去,见到一个闪着蓝色电光的球体-他自称为343妖火(343 Guilty Spark)。


   343妖火告诉队长,他是“第四号基地”的守护者(Monitor),有人擅自释放了洪魔生物(Flood)。造成大混乱。他的职责就是阻止洪魔离开这 个基地和阻止它们的蔓延,但他需要马斯达队长的协助。于是,队长和343妖火再次消失在黄光之中……形势越来越迷雾重重了。




Chapter 7 - The Library 光晕图书馆



   343妖火把马斯达队长带到一个神秘的建筑内部,他称为光晕的图书馆。他带队长来的目的是要取那个有绿色光芒的指令(Index),因为必须用指令激活 光晕,才能制止洪魔的扩散。指令悬浮在一个大洞中间,需要从四楼乘一台大升降机下来才能取得,所以这关的任务就是跟着343妖火一层一层上四楼。


  洪魔不断从各个通风口蜂拥而至,队长打到手都软了,暗自咒骂:你这个混球,到底是想帮我还是想害我啊……


   343妖火说,洪魔放出孢子感染宿主,使有机体发生基因变异,变异了的宿主继续产生孢子,感染其他宿主。洪魔就是靠这种方式繁殖扩散,只要尚存一个未感 染的宿主,洪魔的感染都不会停止。绝不能让洪魔离开基地,否则它们将会不断进化,无法阻止正如所见,那种小洪魔只会爬行,但是感染了有机体之后,人形寄生 体会跑会跳(跳得超高),会使用武器,甚至会驾驶宇宙飞船洪魔,离开光晕……


  在343妖火的呓语中,他说队长现在的战斗服 只相当于“他们”的技术水平的2级,不适合与洪魔抗衡,而“他们”已经是12级了,难道MJORNIR装甲与光晕上的技术有什么渊源?但即使是2级的战斗 服,也具有很好的适应光晕气候改变的能力,因为洪魔会放出毒素破坏光晕大气(记得上一关柯塔娜对大雪天气的评价吗?),一般生物是会受到损害的。


  不久,343妖火会派一队机械卫兵(Sentinel)来对付洪魔,它们发射的死光像切削刀那样切割着洪魔,既可帮你开路,也是省子弹的一个好方法。


   343妖火继续告诉队长一些惊人的事实:原来洪魔危机以前曾经爆发过,“先人(Forerunner)”也曾经平息了上一次灾难。本应该全部消灭它们, 但是,“先人”留下了一些样品以供研究。这个基地就是“先人”专门建造来装载和研究洪魔的。释放洪魔本来是绝对禁止的,某个种族(圣约人?)的过失导致现 在难以收拾的局面,他们必须对此负责……


  最后,队长取得了指令,但马上又被343妖火抢了过去。他解释说那是因为有机体容易被感染,只有机械人不会被感染,而在激活光晕之前,指令绝不允许落到洪魔手中。现在的任务就是马上回到光晕控制室,激活光晕!

Chapter 8 -- Two Betrayals 两个背叛者


   队长和343妖火瞬间转移回控制室,343妖火说,任何有品质和有感知能力的有机体都是洪魔的一个潜在载体。他叫队长把指令插入终端,但奇怪的是,这次 他竟然叫队长为“传教者(Reclaimer)”,仿佛他早已知道队长的职责就是要激活光晕……突然一股无形的推力把343妖火推了开去,柯塔娜出现在终 端上,阻止了队长的行动。


  柯塔娜揭穿了343妖火的诡计原来,“先人”在远古建造了光晕,并用它来装载洪魔。洪魔侵食生 命,银河系所有的生命都可作为它的食物。洪魔的繁殖速度根本无法控制,唯一的阻止洪魔扩散的方法,是断绝它们的食物来源,饿死它们!就是说,激活光晕,并 不会直接消灭洪魔,而是消灭银河系所有的生命!从这种意义上说,光晕的确是一件可怕的武器!马斯达队长差点就成了银河系的千古罪人了。

在 队长质问下,343妖火补充说,一个光晕发射脉冲的最大杀伤半径只有25000光年,这个范围远未达到银河系的范围,但只要其余几个也一起激活的话(老 天,不止一个光晕!),就可完全毁灭整个银河系的生命形式。但是343妖火却反问队长,为何会不了解这些事实?为何忘记了自己以前的职责?似乎在上一次平 息洪魔危机时,队长也是其中的参与者?难道队长与“先人”有某种联系??


  柯塔娜收起了指令,队长取回柯塔娜的晶片,使343妖火无法激活光晕,于是双方马上翻了脸。343妖火留下一队机械卫兵,命令销毁这两个背叛者,但是要保留其首级,自己就消失了。


  马上躲在一块透明挡板后面,避开机械卫兵的死光。对付这种东西,用圣约人的武器比较有效。


  虽然收起了指令,但柯塔娜还是心343妖火有别的办法激活光晕。所以唯一的选择就是毁灭光晕,而且必须赶在洪魔进化到有能力离开光晕之前,把它们一起毁灭掉,否则,地球、银河系、整个宇宙都会难逃厄运。


   柯塔娜有两步计划:第一步,为了拖延343妖火找到第二种办法激活光晕,首先要破坏光晕的三个脉冲发射器,它们隐藏在山谷内部;第二步,为了毁灭光晕, 必须制造一个能量足够大的爆炸,例如一艘太空船的聚变反应堆秋之舰!只要从圣约人的战网中找到秋之舰的坠毁地点,就可以用它来毁灭光晕!


   从控制室出来,这关基本上就是把第五关反过来打一遍(房间中要按地上的箭头的反方向走),不过由于要面对圣约人、洪魔和机械卫兵三种敌人(当然可先等它 们自相残杀),而且圣约人的猎手、鬼魂、女妖、幽灵都纷纷出动,连人形寄生体也进化到会用火箭筒了。所以最好就是逃命,不要浪费子弹了。找到女妖战机 (Banshee)后,马上按照柯塔娜的导航箭头寻找脉冲发射器。

在队长质问下,343妖火补充说,一个光晕发射脉冲的最大杀伤半径只 有25000光年,这个范围远未达到银河系的范围,但只要其余几个也一起激活的话(老天,不止一个光晕!),就可完全毁灭整个银河系的生命形式。但是 343妖火却反问队长,为何会不了解这些事实?为何忘记了自己以前的职责?似乎在上一次平息洪魔危机时,队长也是其中的参与者?难道队长与“先人”有某种 联系??


  柯塔娜收起了指令,队长取回柯塔娜的晶片,使343妖火无法激活光晕,于是双方马上翻了脸。343妖火留下一队机械卫兵,命令销毁这两个背叛者,但是要保留其首级,自己就消失了。


  马上躲在一块透明挡板后面,避开机械卫兵的死光。对付这种东西,用圣约人的武器比较有效。


  虽然收起了指令,但柯塔娜还是心343妖火有别的办法激活光晕。所以唯一的选择就是毁灭光晕,而且必须赶在洪魔进化到有能力离开光晕之前,把它们一起毁灭掉,否则,地球、银河系、整个宇宙都会难逃厄运。


   柯塔娜有两步计划:第一步,为了拖延343妖火找到第二种办法激活光晕,首先要破坏光晕的三个脉冲发射器,它们隐藏在山谷内部;第二步,为了毁灭光晕, 必须制造一个能量足够大的爆炸,例如一艘太空船的聚变反应堆秋之舰!只要从圣约人的战网中找到秋之舰的坠毁地点,就可以用它来毁灭光晕!


   从控制室出来,这关基本上就是把第五关反过来打一遍(房间中要按地上的箭头的反方向走),不过由于要面对圣约人、洪魔和机械卫兵三种敌人(当然可先等它 们自相残杀),而且圣约人的猎手、鬼魂、女妖、幽灵都纷纷出动,连人形寄生体也进化到会用火箭筒了。所以最好就是逃命,不要浪费子弹了。找到女妖战机 (Banshee)后,马上按照柯塔娜的导航箭头寻找脉冲发射器。

柯塔娜不久就找到秋之舰的位置了,但由于接入其聚变反应堆需要得到舰长的权限,所以在去之前还要先找到舰长,获取他的神经晶片。

注意,狙击枪对人形寄生体无效!


  破坏最后一个发射器,柯塔娜找到舰长的位置就在圣约人的那艘“真相和解号”巡洋舰内,而且生命危在旦夕,如果舰长死亡,神经晶片失效,就无法进入秋之舰的反应堆了!坐女妖去已经来不及,所以柯塔娜也用一个瞬间转移,她说是在光晕控制室里面学会的。



Chapter 9 C Keyes 舰长之死



   洪魔占领了圣约人的“真相和解号”巡洋舰,损毁的巡洋舰留出的液体在地面上形成大大小小的池。圣约人非常害怕洪魔驾驶飞船逃离光晕,于是派了一支突击队 想消灭舰上的洪魔,但形势已经不在他们掌握之中了。柯塔娜发现圣约战网一片混乱,但仍然接收到舰长的神经信号,证明神经晶片尚未被破坏。


  队长被传送回巡洋舰的长廊内,但柯塔娜好像没有把瞬间移动学好,把坐标颠倒了……前方一个大坑挡住了去路,只好跳下去(如果不跳而回头的话……)


   现在要找另一条路回到巡洋舰上。碰到圣约人和洪魔混战就等一等,高兴的话可以扔几个手雷过去增加气氛。不过要小心,现在扔出的手雷一般都会引起连环爆 炸。重新上巡洋舰后,发现和第三关的布局大致一样,舰长的神经信号来自飞船的指挥中心。一直杀到指挥中心,却发现舰长……


  洪魔很聪明,它们知道舰长的神经晶片连接着秋之舰,因此只要控制了舰长,就可以控制秋之舰上面所有的飞船,就可以离开光晕!现在,舰长已经不成人形了,一个巨大的绿色肉瘤裹住了他的身体,只突出一张痛苦狰狞的脸奇斯舰长已经死亡,成为洪魔的寄生体了!


  人类已经在这场战争中付出无可估量的牺牲了,绝对不能让洪魔逃离光晕!这是舰长临终前最后的命令了。马斯达队长从舰长的头部挖出神经晶片,现在只剩下最后的任务,回到秋之舰,毁灭光晕!


  最后队长抢了一部女妖火速逃离,消失在夜色之中。


  Chapter 10 C The Maw 绝处逢生



  掠过金光闪闪的海面,马斯达队长和柯塔娜又回到已经破败不堪的秋之舰。


   离家几天,柯塔娜又回到自己的岗位上。秋之舰的反应堆运作良好,于是她启动了自毁程式。突然传来343妖火的笑声!在引擎室中,343妖火查阅着秋之舰 的资料,喃喃自语说找回了他们被遗忘的过去(机械人被创造之前的人类历史?),然后操纵反应堆的控制台,终止了自毁程式……唯今之计,只好直接去破坏反应 堆引擎了。


  大决战会在引擎室进行……四个反应炉都搞定后,引擎已达到临界状态。谁也无法再阻止秋之舰的大爆炸了。柯塔娜建议从秋之舰顶部的维持区逃离。柯塔娜呼叫艾高419在外部接口4C接应。队长跳上一部疣猪战车,随着激动人心的鼓点响起,队长开始在光晕上的最后征途。

离爆炸尚有6分钟时间,只要车技过得去(保持加速,翻车不超过两次,不要管路上的敌人,尽量走没有路障的弯曲通道),按照柯塔娜的指示肯定是够时间的。4C接口是一个圆形平台,在那里等候艾高419接应。

不 幸,艾高419被两部女妖击中坠毁,全员遇难……唯一完好的飞船都没有了,时间无多,怎样逃离光晕呢?柯塔娜意外的发现还有一架长剑战机(Long Sword Fighter)停泊在7号港这是最后的救命稻草了!(怎么有点像《龙珠Z》中,悟空在临爆的那美克星打败菲利之后,找不到飞船逃离的那段情节?)


  时间剩下不到2分钟,伴随着Halo的主题音乐,队长驾车冲到来到一个打开了的停泊港,一堆掩体挡住了去路,队长马上跳车,拼命冲向长剑战机。

洪 魔好像知道乘坐这辆战机可以离开光晕,于是蜂拥而至。队长及时关上了甲板,飞出停泊港,留下无数像留蚁一般的洪魔,留下残存苦苦挣扎的圣约人和人类,被一 个巨大的火球吞噬。(如果用Legendary难度完成游戏,这里还有一段隐藏动画,看完这段动画,你就会明白“真相与和解”的真正含义,加油吧队长!)


  长剑战机飞出了光晕的大气层,在环上出现一团亮光,光晕从那里断裂开,碎片不断扩散、碰撞,伴随着巨大的爆炸。但是,宇宙是真空的,听不到任何声音。


  队长和柯塔娜是唯一的逃出者。为了挽救地球,为了阻止洪魔蔓延,他们消灭了一艘圣约巡洋舰,毁灭了光晕,但也损失了一整支人类部队。柯塔娜认为一切随着光晕的终结而终结,但是,队长却认为新的危机才刚刚开始。


  漆黑的宇宙中,有一点蓝色的亮光在飞行,越来越大,最后,变成一个闪着蓝色电光的球体……(完)

Category: 未分类 | Tags:
6
29
2012
3

openVPN

服务器端

准备工作

检查Tun/Tap是否开通:

cat /dev/net/tun

如果返回内容为: cat: /dev/net/tun: File descriptor in bad stat,表明已经成功启用TUN支持。不然到BurstNET的控制台上打开TUN支持。

 

检查iptables_nat模块是否支持:

iptables -t nat -A POSTROUTING -s 10.8.0.0/24 -o venet0 -j MASQUERADE

如果返回内容为:iptables: Unknown error 4294967295,表明系统还不支持,需要联系客服开通。

 

生成证书

复制一个例子出来:

mkdir ~/easy-rsa
cp /usr/share/openvpn/easy-rsa/2.0 ~/easy-rsa
cd ~/easy-rsa

编辑环境变量vars文件:

# vim vars
export KEY_COUNTRY="CN"
export KEY_PROVINCE="SC"
export KEY_CITY="ChengDu"
export KEY_ORG="FreedomUnion"
export KEY_EMAIL="admin@xiaozhou.net"

导出环境变量:

source vars
./clean-all
./build-ca

生成服务器端证书和密钥,如起名为“morgan-vps”:

./build-key-server morgan-vps

生成客户端证书和密钥(如果给多个人用就生成多个),如起名为“jade”:

./build-key jade

生成证书:

./build-dh

把keys下面的证书文件复制到openvpn目录下:

mkdir /etc/openvpn/keys
cp ~/easy-rsa/keys/*  /etc/openvpn/keys/

编辑配置文件:

# vim /etc/openvpn/server.conf

port 9090
proto tcp
dev tun

ca   /etc/openvpn/keys/ca.crt
cert /etc/openvpn/keys/morgan-vps.crt
key  /etc/openvpn/keys/morgan-vps.key
dh   /etc/openvpn/keys/dh1024.pem 

server 10.8.0.0 255.255.255.0

push "redirect-gateway def1"
push "dhcp-option DNS 8.8.8.8"
push "dhcp-option DNS 8.8.4.4"

client-to-client

keepalive 10 60

comp-lzo

max-clients 10

persist-key
persist-tun

status openvpn-status.log
log-append openvpn.log
verb 3
mute 20

修改:

# sudo vim /etc/sysctl.conf

net.ipv4.ip_forward = 1

 

添加iptables规则:

iptables -t nat -A POSTROUTING -s 10.168.0.0/16 -o eth0 -j MASQUERADE

如果报错: iptables: No chain/target/match by that name  。则改为:

iptables -t nat -A POSTROUTING -s 10.8.0.0/24 -o venet0 -j SNAT --to-source 本机的IP

启动服务:

/usr/sbin/openvpn --config /etc/openvpn/server.conf 

用启用脚本启动VPN与网络:

/etc/init.d/openvpn restart
/etc/init.d/networking restart 

 

 

客户端

复制密钥到配置目录下:

# ls /etc/openvpn/jade-vpn/keys

ca.crt	jade.crt  jade.key

配置文件:

# cat /etc/openvpn/jade-vpn/client.ovpn

client
dev tun
proto tcp
remote 服务器IP 服务器端口
resolv-retry infinite
nobind
persist-key
persist-tun
ca /etc/openvpn/jade-vpn/keys/ca.crt
cert /etc/openvpn/jade-vpn/keys/jade.crt
key  /etc/openvpn/jade-vpn/keys/jade.key
ns-cert-type server
comp-lzo
verb 3

可以建立一个启动脚本:

# vim jade-vpn.sh

#!/bin/bash
sudo openvpn --config /etc/openvpn/jade-vpn/client.ovpn

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

Category: linux | Tags:
6
27
2012
10

系统安装备忘

备份

备份没有在github上存着的东西:

offlineimap

msmtprc

pidgin聊天记录

firefox收藏夹

等等……继续补充……

 

安装系统时别手贱“下一步”“下一步”。很多安装向导默认会“使用*整个*硬盘”!

重新分区时,再三确认上面的内容已经备份。想想某年某月某日错把备份完数据的U盘做安装盘的二逼事迹。

 

安装

好像一时还想不到什么话说……

 

配置

locale

安装locales程序:

apt-get install locales  

然后配置所用的语系:

dpkg-reconfigure locales 

中文常用的的locale:

  • en_US.ISO-8859-1
  • en_US.UTF-8
  • zh_CN.GB2312
  • zh_CN.GB18030
  • zh_CN.UTF-8
  • zh_CN.GBK
  • zh_TW.BIG5
  • zh_TW.UTF-8

缺省locale为en_US.utf8,

这样就完成了,可以查看一下中的语系:

locale -a 

 

用户

添加常用账号,默认会建立同名组

groupadd user001
useradd user001 -g user001 -d /home/user001 -s /usr/bin/zsh 

修改密码

passwd user001

建立用户目录,别忘记修改权限,不然什么东西都被人家看到……(羞)

mkdir /home/user001
chgrp user001 /home/user001 
chown user001 /home/user001
chmod 700 /home/user001

同步用户和组(不过不是直接修改配置文件的话应该用不着)

grpconv

再提醒一下自己以后删除用户的时候不要-r参数忘记删除home目录和mail。 还有userdel会把用户的组也一块删除掉,当心当心

userdel -r user001

用于天朝特色用途的账号,没有登录的必要。不给shell:

useradd user001 -g user001 -s /bin/false


sudo

安装sudo

apt-get install sudo

编辑/etc/sudoers增加sudo权限

user001 ALL=(ALL) ALL

更详细的配置介绍:

1、别名设置
别名主要分成4种,分别是:
1)Host_Alias 主机别名,就是主机的列表
如:Host_Alias HOST_FLAG = hostname1, hostname2, hostname3


2)Cmnd_Alias 命令别名,就是允许执行的命令的列表
如:Cmnd_Alias COMMAND_FLAG = command1, command2, command3


3)User_Alias 用户别名,就是具有sudo权限的用户的列表
如:User_Alias USER_FLAG = user1, user2, user3


4)Runas_Alias Runas别名,就是用户以什么身份执行(例如root,或者oracle)的列表
如:Runas_Alias RUNAS_FLAG = operator1, operator2, operator3
别名格式是:Alias_Type NAME = item1, item2, ……

2、权限设置
首先看看授权规则:
格式: 授权用户 主机 = [(目的用户)] [NOPASSWD:] 命令列表
如:tony ALL=(ALL) NOPASSWD:ALL
其中NOPASSWD是指不需要密码验证

例子:

# groups
User_Alias  ROOT = user1, user2, user3
User_Alias  WEBMASTERS = user4, user5, user6
 
# commands
Cmnd_Alias  APACHE = /usr/local/sbin/kickapache
Cmnd_Alias  TAIL = /usr/bin/tail
Cmnd_Alias      SHUTDOWN = /sbin/shutdown
Cmnd_Alias      APT = /usr/bin/apt-get, /usr/bin/dpkg
 
# privileges 
ROOT        ALL = (ALL) ALL
WEBMASTERS  ALL = PASSWD : APACHE, TAIL
admin       ALL = NOPASSWD : /etc/init.d/apache

参数:
-l 显示出自己(执行 sudo 的使用者)的权限
-v 因为 sudo 在第一次执行时或是在 N 分钟内没有执行(N 预设为五)会问密码,这个参数是重新做一次确认,假如超过 N 分钟,也会问密码
-k 将会强迫使用者在下一次执行 sudo 时问密码(不论有没有超过 N 分钟)
-b 将要执行的指令放在后台执行
-p prompt 能够更改问密码的提示语,其中 %u 会代换为使用者的帐号名称, %h 会显示主机名称
-u username/#uid 不加此参数,代表要以 root 的身份执行指令,而加了此参数,能够以 username 的身份执行指令(#uid 为该 username 的使用者号码)
-s 执行环境变量中的 SHELL 所指定的 shell ,或是 /etc/passwd 里所指定的 shell
-H 将环境变数中的 HOME (家目录)指定为要变更身份的使用者家目录(如不加 -u 参数就是系统管理者 root )

 

安全设置

SSH

修改SSH端口,禁止root远程登录

# vi /etc/ssh/sshd_config
Port 1234
PermitRootLogin no

重启服务

service sshd restart

生成登录用的密钥

ssh-keygen -t rsa

把公钥上传到服务器

cat  ~/.ssh/id_rsa.pub | ssh user001@192.168.1.1 "cat - >> ~/.ssh/authorized_keys"

如果密钥中设置了passphrase,则需要输passphrase登录服务器。为了更方便可以通过ssh-agent来帮助修改"~/.ssh/id_rsa"文件。看起来像是自动输入passphrase(只是而已):

ssh-add

备注:对于SSH2兼容格式的公钥,可以转换成为Openssh兼容格式

ssh-keygen -i -f Identity.pub >> /root/.ssh/authorized_keys2

禁止密码登录,只允许key登录:还不知道怎么搞~OTZ

iptables

清除已有的规则:

iptables -F
iptables -X
iptables -Z

开放常用的端口:

# 允许本地回环接口
iptables -A INPUT -s 127.0.0.1 -d 127.0.0.1 -j ACCEPT
# 放行已经连接的相关连接
iptables -A INPUT -m state --state ESTABLISHED,RELATED -j ACCEPT
# 不限制外出
iptables -A OUTPUT -j ACCEPT
# 放行常用入口请求 ssh http ftp
iptables -A INPUT -p tcp --dport 22 -j ACCEPT
iptables -A INPUT -p tcp --dport 80 -j ACCEPT
iptables -A INPUT -p tcp --dport 21 -j ACCEPT
iptables -A INPUT -p tcp --dport 20 -j ACCEPT
# 同样格式的其他入口
# 禁止其他访问入口,注意别把ssh的22端口给禁了
iptables -A INPUT -j REJECT
iptables -A FORWARD -j REJECT

# 封一个IP
iptables -I INPUT -s 123.45.6.7 -j DROP
# 封123.0.0.1~123.255.255.254整个段
iptables -I INPUT -s 123.0.0.0/8 -j DROP
# 封123.45.0.1到123.45.255.254
iptables -I INPUT -s 124.45.0.0/16 -j DROP
# 封123.45.6.1到123.45.6.254
iptables -I INPUT -s 123.45.6.0/24 -j DROP

检查已经添加的规则:

iptables -L -n --line-numbers

可以按显示的chain类与行号删除一条规则,如 INPUT中的第3条:

iptables -D INPUT 3

网卡启动时加载规则:

/etc/network/if-pre-up.d/iptables
#!/bin/bash
iptables-restore < /etc/iptables.rules
chmod +x /etc/network/if-pre-up.d/iptables

网卡关闭时保存规则

/etc/network/if-post-down.d/iptables
#!/bin/bash
iptables-save > /etc/iptables.rules
chmod +x /etc/network/if-post-down.d/iptables

 

iptables-persistent

这是debian内用于iptables规则持久化的工具,你可以编辑/etc/iptables/rules.v4来修改防火墙规则。一般来说,至少要包含以下内容:

-A INPUT -m state –state RELATED,ESTABLISHED -j ACCEPT
-A INPUT -i lo -j ACCEPT
-A INPUT -i tun+ -j ACCEPT
-A INPUT -i ppp+ -j ACCEPT
-A INPUT -p tcp -m multiport –dport 22,xxx,xxx,xxx -j ACCEPT
-A INPUT -p udp -m multiport –dport xxx,xxx,xxx -j ACCEPT

强烈建议先保存一个没问题的iptables,然后直接修改iptables,再保存。这样当改错了导致无法管理的时候,只要重启就可以恢复vps工作。

 

denyhosts

python编写用来踢掉试图尝试ssh密码的ip。如果已经用了通过key的连接方式,你可以一次就直接踢掉对方ip。

 

常用软件

jre

sudo apt-get install python-software-properties
add-apt-repository "deb http://archive.canonical.com/ lucid partner"
sudo apt-get update
sudo apt-get install openjdk-6-jre
sudo apt-get install openjdk-6-jdk

mocp

sudo apt-get install moc moc-ffmpeg-plugin 

增加配置文件

vim ~/.moc/config

内容:

XTermTheme = nightly_theme # 背景透明
ReadTags   = no            # 中文歌名乱码





 

网络管理

ifstat

ifstat是用于网络流量管理的工具,可以告诉你网络目标的流量是多少。
 

dnsutils

dnsutils里面包含了不少用于管理dns的工具,包括我们常用的nslookup,还有相对少用的dig。
 

mtr-tiny

mtr是一个traceroute工具,比后者好用很多。这个工具可以快速跟踪路由。
 

vnstat

vnstat是用于跟踪网卡流量的工具,尤其对于每个月都有限额的vps,这个工具更有意义。注意安装完成后需要初始化每个网卡,然后重启服务,而不是马上能够工作。

 

网络服务

pptp

pptp是一个经典的vpn服务,直接安装pptpd就好。注意,部分手机不支持128bit的mppe,关闭后可以连接。但是windows只支持128bit的mppe,关掉就无法连接。So,自己权衡。
 
openvpn
 
openpn是一个非常稳定而强大的vpn程序,他使用udp作为连接协议。其实openvpn有tcp协议模式,但是速度比udp慢很多。openvpn的配置可以参考贝壳童鞋的文章(反正很本文很多工具都是从他blog上学来的):
1.搭建家用的OpenVPN服务器:http://shell909090.com/blog/2009/09/%E6%90%AD%E5%BB%BA%E5%AE%B6%E7%94%A8%E7%9A%84openvpn%E6%9C%8D%E5%8A%A1%E5%99%A8/
2.说说x509证书链:http://shell909090.com/blog/2011/04/%E8%AF%B4%E8%AF%B4x509%E8%AF%81%E4%B9%A6%E9%93%BE/
3.再论openvpn的搭建:http://shell909090.com/blog/2011/05/%E5%86%8D%E8%AE%BAopenvpn%E7%9A%84%E6%90%AD%E5%BB%BA/
 

ssh
 
ssh用于翻墙常见两种模式,固定端口转发和动态端口转发。前者使用-R将远程的某个端口映射到本地。通常而言,映射的都是squid或者polipo(推荐后者,内存消耗更小,更好配置)。这样相当于在 本地可以访问远程的代理,从而达到翻墙的效果。
命令:
ssh -L port:localhost:port …
 
而动态端口转发则是使用ssh -D port …,将本地的port端口变成一个支持socks5协议的代理服务器。相比而言,-D模式更加灵活,提供了全协议的访问, 本地可以通过polipo转换为http代理。而-L模式则不能提供socks5代理功能(除非远程的端口上是socks5代理服务,但是这样就回到了 -D模式,反而多开了一个服务)。但是有些时候(例如android的ssh翻墙软件)只支持后者的模式。另外,不要用日常管理帐号翻墙。新开一个翻墙帐号,并且设定独立的key。然后禁用shell,在ssh的时候,使用参数-CNq,这个参数可以不打开shell。如果网络不稳定,可以加上-o ServerAliveInterval 30。
 
stunnel
stunnel本身没有任何功效,他只是将你的普通连接转换为ssl连接而已。当这个程序搭配其他程序,例如polipo,就可以实现一个ssl级别的代理。
 
httptunnel
这是一个服务软件,服务器端运行一个httptunnel,客户端运行一个。而后客户端就可以获得一个到服务器端的tcp连接,不受限的。
 
polipo
polipo常见有两种模式,端口转发模式和ssl模式。两者都在前文有说。端口转发模式配合ssh用,ssl模式配合stunnel用。
以上的服务看似很多,实际上,在128M内存的实例上完全可以运行其中大部分的服务。你可以在一台服务器上运行其中多个,以保证全天候的服务。

 

 

Category: linux | Tags:
6
4
2012
0

debian包管理

1.dpkg包管理工具  
dpkg --info "软件包名" --列出软件包解包后的包名称.  
dpkg -l --列出当前系统中所有的包.可以和参数less一起使用在分屏查看.  
dpkg -l |grep -i "软件包名" --查看系统中与"软件包名"相关联的包.  
dpkg -s 查询已安装的包的详细信息.  
dpkg -L 查询系统中已安装的软件包所安装的位置.  
dpkg -S 查询系统中某个文件属于哪个软件包.  
dpkg -I 查询deb包的详细信息,在一个软件包下载到本地之后看看用不用安装(看一下呗).  
dpkg -i 手动安装软件包(这个命令并不能解决软件包之前的依赖性问题),如果在安装某一个软件包的时候遇到了软件依赖的问题,可以用apt-get -f install在解决信赖性这个问题.  
dpkg -r 卸载软件包.不是完全的卸载,它的配置文件还存在.  
dpkg -P 全部卸载(但是还是不能解决软件包的依赖性的问题)  
dpkg -reconfigure 重新配置  


2. apt包管理工具
(1)GTK图形的"synaptic",这是APT的前端工具.  
(2)"aptitude",这也是APT的前端工具.  
用APT管理工具进行包的管理,可以有以下几种方法做源:  
(1)拿安装盘做源,方法如下:  
apt-cdrom ident 扫描光盘的信息  
apt-cdrom add 添加光盘源  
(2)这也是最常用的方法就是把源添加到/etc/apt/source.list中,之后更新列apt-get update  
APT管理工具常用命令  
apt-cache 加上不同的子命令和参数的使用可以实现查找,显示软件,包信息及包信赖关系等功能.  
apt-cache stats 显示当前系统所有使用的Debain数据源的统计信息.  
apt-cache search +"包名",可以查找相关的软件包.  
apt-cache show +"包名",可以显示指定软件包的详细信息.  
apt-cache depends +"包名",可以查找软件包的依赖关系.  
apt-get upgrade 更新系统中所有的包到最新版  
apt-get install 安装软件包  
apt-get --reindtall install 重新安装软件包  
apt-get remove 卸载软件包  
apt-get --purge remove 完全卸载软件包  
apt-get clean 清除无用的软件包  
在用命令apt-get install之前,是先将软件包下载到/var/cache/apt/archives中,之后再进行安装的.所以我们可以用apt-get clean清除/var/cache/apt/archives目录中的软件包.  
源码包安装  
apt-cache showsrc 查找看源码包的文件信息(在下载之前)  
apt-get source 下载源码包.  
apt-get build-dep +"包名" 构建源码包的编译环境. 


3.apt-get与dpkg的一些基本用法  
apt-get install packagename #安装一个新软件包  
apt-get remove packagename #卸载一个已安装的软件包(保留配置文件)  
apt-get --purge remove packagename #卸载一个已安装的软件包(删除配置文件)  
dpkg --force-all --purge packagename #强制卸载,风险大!  
apt-get upgrade #更新所有已安装的软件包  
apt-get dist-upgrade #将系统升级到新版本  
apt-get clean #清理所有软件缓存  
apt-get autoclean #清理旧版本的软件缓存  
apt-get autoremove #删除系统不再使用的孤立软件  
apt-cdrom add #增加一个光盘源  
auto-apt run ./configure #编译时缺少h文件的自动处理  
apt-cache search 正则表达式 #在软件包列表中搜索字符串  
dpkg -l 正则表达式 #列出所有与模式相匹配的软件包  
dpkg -l |grep ^rc|awk '{print $2}' |  #xargs dpkg -P #清除所有已删除包的残馀配置文件  
dpkg -i, --install XXX.deb #安装XXX.deb软件包:dpkg --install stardict_3.0.1-1_i386.deb  
dpkg -r, --remove, -P, --purge package...|-a|--pending #删除一个软件包:dpkg -r stardict

Category: linux | Tags: apt debian dpkg
4
19
2012
0

失窃追踪

笔记本电脑被盗后的追踪工具:

http://preyproject.com/

 

但是我想,应该只有对mac用户有用。如果你电脑里的windows或是linux操作系统的话。人家做的第一件事情就是重装系统了……

Category: 其他 | Tags: 失盗追踪

Host by is-Programmer.com | Power by Chito 1.3.3 beta | Theme: Aeros 2.0 by TheBuckmaker.com