《快学Scala》(英文版:《Scala for the Impatient》),代码已传github:
https://github.com/vernonzheng/scala-for-the-Impatient
书为第一版。scala为2.11.4,jdk1.7.45,操作系统Mac OS X Yosemite 10.10.1。
第八章 继承
8.1
扩展如下的BankAccount类,新类CheckingAccount对每次存款和取款都收取1美元的手续费
class BankAccount(initialBalance:Double){
private var balance = initialBalance
def deposit(amount:Double) = { balance += amount; balance}
def withdraw(amount:Double) = {balance -= amount; balance}
}
1 2 3 4 5 6 7 8 9 10
| class BankAccount(initialBalance:Double){ private var balance = initialBalance def deposit(amount:Double) = { balance += amount; balance} def withdraw(amount:Double) = {balance -= amount; balance} } class CheckingAccount(initialBanlance:Double) extends BankAccount(initialBanlance){ override def deposit(amount:Double) = super.deposit(amount-1) override def withdraw(amount:Double) = super.withdraw(amount+1) }
|
8.2
扩展前一个练习的BankAccount类,新类SavingsAccount每个月都有利息产生(earnMonthlyInterest方法被调用),并且有每月三次免手续费的存款或取款。在earnMonthlyInterest方法中重置交易计数。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| class BankAccount(initialBalance:Double){ private var balance = initialBalance def deposit(amount:Double) = { balance += amount; balance} def withdraw(amount:Double) = {balance -= amount; balance} } class SavingsAccount(initialBalance:Double) extends BankAccount(initialBalance){ private var num:Int = _ def earnMonthlyInterest()={ num = 3 super.deposit(1) } override def deposit(amount: Double): Double = { num -= 1 if(num < 0) super.deposit(amount - 1) else super.deposit(amount) } override def withdraw(amount: Double): Double = { num -= 1 if (num < 0) super.withdraw(amount + 1) else super.withdraw(amount) } }
|
8.3
翻开你喜欢的Java或C++教科书,一定会找到用来讲解继承层级的实例,可能是员工,宠物,图形或类似的东西。用Scala来实现这个示例。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| /** * java * class Art{ Art(){System.out.println("Art constructor");} } class Drawing extends Art{ Drawing() {System.out.println("Drawing constructor");} } public class Cartoon extends Drawing{ public Cartoon() { System.out.println("Cartoon constructor");} } */ class Art{ println("Art constructor") } class Drawing extends Art{ println("Drawing constructor") } class Cartoon extends Drawing{ println("Cartoon constructor") }
|
8.4
定义一个抽象类Item,加入方法price和description。SimpleItem是一个在构造器中给出价格和描述的物件。利用val可以重写def这个事实。Bundle是一个可以包含其他物件的物件。其价格是打包中所有物件的价格之和。同时提供一个将物件添加到打包当中的机制,以及一个适合的description方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
| import collection.mutable.ArrayBuffer abstract class Item{ def price():Double def description():String override def toString():String={ "description:" + description() + " price:" + price() } } class SimpleItem(val price:Double,val description:String) extends Item{ } class Bundle extends Item{ val items = new ArrayBuffer[Item]() def addItem(item:Item){ items += item } def price(): Double = { var total = 0d items.foreach(total += _.price()) total } def description(): String = { items.mkString(" ") } }
|
8.5
设计一个Point类,其x和y坐标可以通过构造器提供。提供一个子类LabeledPoint,其构造器接受一个标签值和x,y坐标,比如:new LabeledPoint(“Black Thursday”,1929,230.07)
1 2
| class Point(x:Double, y:Double) class LabeledPoint(x:Double, y:Double, tag:String) extends Point(x,y)
|
8.6
定义一个抽象类Shape,一个抽象方法centerPoint,以及该抽象类的子类Rectangle和Circle。为子类提供合适的构造器,并重写centerPoint方法
1 2 3 4 5 6 7 8 9 10 11
| abstract class Shape{ def centerPoint() } class Rectangle(startX:Int,startY:Int,endX:Int,endY:Int) extends Shape{ def centerPoint() {} } class Circle(x:Int,y:Int,radius:Double) extends Shape{ def centerPoint() {} }
|
8.7
提供一个Square类,扩展自java.awt.Rectangle并且是三个构造器:一个以给定的端点和宽度构造正方形,一个以(0,0)为端点和给定的宽度构造正方形,一个以(0,0)为端点,0为宽度构造正方形
1 2 3 4 5 6 7 8 9 10 11 12
| import java.awt.{Point, Rectangle} class Square(point:Point,width:Int) extends Rectangle(point.x,point.y,width,width){ def this(){ this(new Point(0,0),0) } def this(width:Int){ this(new Point(0,0),width) } }
|
8.8
编译8.6节中的Person和SecretAgent类并使用javap分析类文件。总共有多少name的getter方法?它们分别取什么值?(提示:可以使用-c和-private选项)
总共两个。Person中取得的是传入的name,而SecretAgent中取得的是默认的”secret”
8.9
在8.10节的Creature类中,将val range替换成一个def。如果你在Ant子类中也用def的话会有什么效果?如果在子类中使用val又会有什么效果?为什么?
在Ant中使用def没有问题。但是如果使用val则无法编译。因为val只能重写不带参数的def。这里的def是带参数的
8.10
文件scala/collection/immutable/Stack.scala包含如下定义:
class Stack[A] protected (protected val elems: List[A])
请解释protected关键字的含义。(提示:回顾我们在第5章中关于私有构造器的讨论) 此构造方法只能被其子类来调用,而不能被外界直接调用
此构造方法只能被其子类来调用,而不能被外界直接调用
参考:
《快学Scala》:http://book.douban.com/subject/19971952/
(转载本站文章请注明作者和出处 Vernon Zheng(郑雪峰) – vernonzheng.com ,请勿用于任何商业用途)