Scala之OOP讲解

Scala OOP

前序

Scala 为纯粹OOP
	1、不支持基本类型:一切皆为对象 Byte,Int,...
	2、不支持静态关键字:static	
	3、支持类型推断【通过判断泛型的父子关系来确定泛型类的父子关系=>协变,逆变,不变】和类型预定,	   动静结合

java不是纯粹的OOP的主要原因支持基本类型,支持静态关键字。

1、类

1.1:类【主构造器】的基本构成

/*
	关键字:class
	创建对象:new
	内含:属性(成员变量)和方法(函数=>解决问题,方法=>用于设计。因此在类中推荐写方法)
	与java区别:
		1、默认访问修饰符为 public,也支持 private 和 protected
		2、没有构造方法,通过构造参数列表实现对象创建。因为scala类本身就是【主】构造器
*/

// 主构造器
class Point(x:Int,y:Int) {
  // 【属性】: 构造参数名和属性不能同名
  private var _x:Int = x
  private var _y:Int = y

  // 【方法】
  def move(offsetX:Int,offsetY:Int): Unit = {
    _x += offsetX
    _y += offsetY
  }
    
  //【获取属性值】
  def getX = _x
  def getY = _y

  //【修改属性值】
  def setX(x:Int) = {_x=x}
  def setY(y:Int) = {_y=y}

  // 【重写】
  override def toString: String = s"Point{x=${_x},y=${_y}}"
}

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    //创建对象
    val point = new Point(1, 2)

    //方法调用
    point.move(2,4)
    println(point)
      
    //获取属性值
    println(point.getX)
      
    //修改属性值
    point.setX(21)
    println(point)
  }
}
------------------------
Point{x=3,y=6}
2
Point{x=21,y=10}
------------------------

访问修饰符

修饰符类(class)伴生对象(object)子类(subclass)同包(package)全局(world)
default(public)YYYYY
protectedYYYNN
privateYYNNN

1.2:辅助构造器

辅助构造器:基于主构造器的重载

class Point(x:Int,y:Int) {
  // 属性: 构造参数名和属性不能同名
  private var _x:Int = x
  private var _y:Int = y
  
  //辅助构造器【必须调用主构造器,名字只能是this】
  def this(x:Int) = this(x,10)

  // 重写方法
  override def toString: String = s"Point{x=${_x},y=${_y}}"
}

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    //创建对象
    val point = new Point(2)
    println(point)
  }
}
------------------------
Point{x=2,y=10}
------------------------

2、继承

class Point(x:Int,y:Int) {
  // 属性: 构造参数名和属性不能同名
  private var _x:Int = x
  private var _y:Int = y

  // 方法
  def move(offsetX:Int,offsetY:Int): Unit = {
    _x += offsetX
    _y += offsetY
  }

  //获取属性值
  def getX = _x
  def getY = _y

  //修改属性值
  def setX(x:Int) = {_x=x}
  def setY(y:Int) = {_y=y}

  //辅助构造器
  def this(x:Int) = this(x,10)

  // 重写方法
  override def toString: String = s"Point{x=${_x},y=${_y}}"
}

/**
 * 继承: extends & override
 *	  目的:扩展【继承父类的主构造器】
 *    	ColorPoint中的x,y会调用Point的x,y
 *    	colorPoint会继承父类Point的所有内容
 *	  注意:私有属性无法被继承
 */
class ColorPoint(var x:Int,var y:Int,color:String) extends Point(x, y){
  //扩展属性
  private var _color:String = color

  //重写move方法:将x与y变为var变量【默认传入的是常量val】
  override def move(offsetX: Int, offsetY: Int): Unit = {
    x += offsetX*2
    y += offsetY*2
    println(s"point moved to {${x},${y}}")
  }

  //多态
  override def toString: String = s"{POINT: ${super.toString},COLOR: ${_color}}"
}

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    val colorPoint = new ColorPoint(10, 20, "red")
    colorPoint.move(2,4)
    println(colorPoint)
  }
}
----------------------------------------
point moved to {14,28}
{POINT: Point{x=10,y=20},COLOR: red}
----------------------------------------

3、抽象类

3.1:抽象类基本知识点

抽象类: <=> 相当于java接口
	1.关键字:abstract & extends & override
	2.抽象类中可以有抽象方法(没有方法体的方法即抽象方法),也可以存在普通方法
	3.无法实例化,需要在子类实体化后才可被使用【可以有构造器】
	4.使用 abstract 关键字修饰
	5.override5.1:子类重写父类【抽象】方法可以省略 override 关键字,但不推荐省略
		5.2:子类重写父类【非抽象】方法必须写 override 关键字

3.2:抽象类详细案列讲解

//抽象类:必须用 abstract 修饰
abstract class Shape {
  // 抽象类中的抽象方法无需 abstract 修饰,因为【没有方法体的方法】默认为【抽象方法】
  def draw():Unit // 抽象方法
    
  // 可以包含普通方法
  def show = println("this is shape")//普通方法(非抽象方法)
}

class Circle(radius:Float) extends Shape{
  private var _radius:Float = radius

  //重写抽象类中的【抽象方法】【必须】,可以并推荐添加 override 关键字
  override def draw(): Unit = println("you are drawing Circle...")

  //重写抽象类中的【非抽象方法】【可选】,必须添加 override 关键字
  override def show: Unit = println(s"it is a Circle with radius of ${_radius} CM")

}

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    val circle = new Circle(12.56f)
    circle.show
    circle.draw()
  }
}
-------------------------------------------
this is a Circle with radius of 12.56 CM
you are drawing Circle...
-------------------------------------------

4、单例对象

4.1:基本知识点

单列对象:
	代替 Java 中的 static 关键字,作为【工具方法】来调用
		1、关键字:object
		2、可以包含【属性】和【方法】,且可以通过单例对象名直接调用
		3、采取【惰性模式】,第一次被访问时创建
		4、【无构造器,且不能 new5、程序入口方法main必须定义在单例对象中
		6、【同一个文件】中【同名的类和单例对象】形成【绑定关系】,并称之为【伴生类和伴生对象】

4.2:单列对象

//单列对象
object Util{
  //属性
  val PI:Float = 3.14f
  var count:Int = 0

  //方法
  def resume = println("it is a utility object")
}

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    println(Util.PI)

    Util.count += 5
    println(Util.count)

    Util.resume
  }
}
------------------------
3.14
5
it is a utility object
------------------------

4.3:为什么要用单列对象?

单列对象是唯一的,具有资源共享特性【资源对于所有类的对象是共享的】。因此,类中共性的内容写在单列对象中,实现资源共享,避免重复创建问题。差异化的内容写在类中

4.4:伴生类与伴生对象【引申】

伴生对象是一种特殊的单列对象伴生对象属于单列对象的一种。伴生对象与伴生类之间可以自由访问。

4.4.1:【伴生对象】中可以自由访问【伴生类】中所有资源

apply和unapply讲解

package scala.cha06
// 伴生类
class Commodity(sku:String,price:Float) {
  private var _sku:String = sku
  private var _price:Float = price

  def getSku = _sku
  def getPrice = _price
}
// 伴生对象
object Commodity{
  //apply【包裹】:实现new的简化操作【参数传入,返回对象】
  def apply(sku: String, price: Float): Commodity = new Commodity(sku, price)

  //unapply【拆解】:【伴生对象】中可以自由访问【伴生类】中所有资源
  def unapply(arg: Commodity):(String, Float) = (arg._sku,arg._price)
}
===================================================================================
package scala.cha06
object ScalaOOP {
  def main(args: Array[String]): Unit = {
    //不new返回单列对象apply;new返回伴生类 => 效果一致
    // 自动调用 apply 完成 Commodity 对象构造 
    val commodity: Commodity = Commodity("草莓", 5.8f)
    println(commodity.getSku)
    println(commodity.getPrice)

    //unapply
    val t2: (String, Float) = Commodity.unapply(commodity)
    println(t2._1)
    println(t2._2)
  }
}
-----------------
草莓
5.8
草莓
5.8
-----------------
4.4.2:【伴生类】自由访问【伴生对象】内的资源
package scala.cha06
// 伴生类
class Commodity(sku:String,price:Float) {
  private var _sku:String = sku
  private var _price:Float = price //原价

  def getSku = _sku
  def getPrice = _price //原价
    
  //伴生类中自由访问伴生对象的所有资源
  def getSalePrice = discount(_price) //折扣价
}
// 伴生对象
object Commodity{
  private var _discount:Float = 1 //商品折扣

  //apply:实现new的简化操作【参数传入,返回对象】
  def apply(sku: String, price: Float): Commodity = new Commodity(sku, price)

  //自定义方法
  def setDiscount(discount:Float) = {_discount = discount} //设定折扣
  def discount(price:Float) = price * _discount
}
===================================================================================
package scala.cha06

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    Commodity.setDiscount(0.88f)
    val commodity: Commodity = Commodity("草莓", 5.8f)//单列对象
    println(commodity.getSalePrice)
  }
}
--------------
5.104
--------------

4.5:小练习

4.5.1:题目要求
1、准备工作
    假设类Book有属性title和author(多个),books是Book的列表
    实现Book类,同时使用主构造器与辅助构造器
    实现Book的伴生对象,使用伴生对象创建Book实例
    创建books,使用List[Book]初始化5个以上Book实例
2、功能实现:
    1.找出books中书名包含“xxx”的书,并打印书名
    2.找出books中作者名以“xxx”打头的书,并打印书名
4.5.2:Book类与对象的构建
package scala.cha06
// 伴生类
class Book(title:String,author:Array[String]) {
  private var _title:String = title
  private var _authors:Array[String] = author

  //辅助构造器:实现类型转换
  def this(title:String,author:String) = this(title,Array(author))

  // 设计业务:解决问题前置化(在源头解决问题)
  //1.判断书名是否包含“xxx”的方法
  def titleContains(sub:String) = _title.contains(sub)
  //2.判断作者名是否以“xxx”打头的方法
  def authorContains(name:String) = { _authors.count(_.startsWith(name))>0 }

  def getTitle = _title

  override def toString: String = s"$title\t${_authors.mkString(",")}"
}
// 伴生对象
object Book{
  //主构造器
  def apply(title: String, author: Array[String]): Book = new Book(title, author)
  //辅助构造器
  def apply(title: String, author: String): Book = new Book(title, author)
}
4.5.3:功能实现
package scala.cha06

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    val list = List(
      // 书名,作者名
      Book("武侠:最强小保安",Array("张三","李四","王五")),
      Book("都市:上门赘婿",Array("阿强","洞冥福","花花")),
      Book("武侠:翔龙会",Array("阿庆嫂","黄世仁")),
      Book("都市:缘起",Array("徐世明","张丘月")),
      Book("武侠:小李飞刀",Array("王栋","李宏","张明")),
    )

    //1.找出books中书名包含“xxx”的书,并打印书名
    list
      .collect({
        case book if book.titleContains("武侠") => book.getTitle
      }).foreach(println)

    println("===============")

    //2.找出books中作者名以“xxx”打头的书,并打印书名
    list
      .collect({
        case book if book.authorContains("李") => book.getTitle
      }).foreach(println)
  }
}
------------------------
武侠:最强小保安
武侠:翔龙会
武侠:小李飞刀
===============
武侠:最强小保安
武侠:小李飞刀
------------------------

5、特质

5.1:特质基本知识点

特质: <=> 【相当于java中接口】
	1、关键字:trait & extends & with & override
	2、包含字段、方法,亦可包含字段和方法的实现
	3、类、单例对象、普通对象【都可以扩展】特质
	4、不能实例化,【没有构造器】
	5、单根继承extends,借助 with 实现多混入
		5.1extendswith区分:
			is a : extends 继承性扩展【只能用一次】
			has a : implements|with 功能性扩展【可使用多次】

5.2:特质详细案例讲解

trait Animal{
  //抽象属性
  var name:String
  //普通属性
  val TYPE:String = "动物"

  //抽象方法
  def roar:Unit
  //普通方法
  def me = s"$TYPE:$name"
}

//功能性trait
trait BySwing{
  def fly():Unit // 飞行
}
trait ByFoot{
  def walk():Unit // 走路
  def run():Unit // 跑步
}
trait ByTail{
  def swim():Unit // 游泳
}

/**
 * 1.单根继承(Animal与Cat之间是is a的关系【Animal是主要特质trait】)
 * 2.with实现多混入(ByFoot,ByTail与Cat之间是has a的关系【ByFoot,ByTail额外特质trait】)
 */
//猫
class Cat(nickName:String) extends Animal with ByFoot with ByTail {
  override var name: String = nickName

  override def roar: Unit = println(s"猫${name}喵喵...")
  override def walk(): Unit = println(s"猫${name}悠闲地漫步...")
  override def run(): Unit = println(s"猫${name}疯狂的奔跑...")
  override def swim(): Unit = println(s"猫${name}水里游泳...")
}

object TraitTest {
  def main(args: Array[String]): Unit = {
    val animal = new Cat("Tom")
    println(animal.name)
    animal.roar
    animal.run()
    animal.walk()
    animal.swim()
  }
}
---------------------
Tom
猫Tom喵喵...
猫Tom疯狂的奔跑...
猫Tom悠闲地漫步...
---------------------

5.2:优化特质案列

package scala.cha06

//特质trait无法使用构造器,因此用abstract来代替
abstract class Animal(brand:String){
  //抽象属性
  var name:String
  //普通属性
  var _type:String = brand

  //抽象方法
  def roar:Unit
  //普通方法
  def me = s"${_type}:$name"
}

//功能性trait
trait BySwing{
  def fly():Unit // 飞行
}
trait ByFoot{
  def walk():Unit // 走路
  def run():Unit // 跑步
}
trait ByTail{
  def swim():Unit // 游泳
}

/**
 * 1.单根继承(Animal与Cat之间是is a的关系【Animal是主要特质trait】)
 * 2.with实现多混入(ByFoot与Cat之间是has a的关系【额外特质trait】)
 */
//猫
class Cat(nickName:String) extends Animal("猫") with ByFoot with ByTail {
  override var name: String = nickName

  override def roar: Unit = println(s"${_type}${name}喵喵...")
  override def walk(): Unit = println(s"${_type}${name}悠闲地漫步...")
  override def run(): Unit = println(s"${_type}${name}疯狂的奔跑...")
  override def swim(): Unit = println(s"${_type}${name}水里游泳...")
}
//鸟
class Bird(nickName:String) extends Animal("鸟") with BySwing with ByFoot{
  override var name: String = nickName

  override def roar: Unit = println(s"${_type}${name}叽叽喳喳的叫...")
  override def fly(): Unit = println(s"${_type}${name}天空翱翔...")
  override def walk(): Unit = println(s"${_type}${name}闲庭信步...")
  override def run(): Unit = println(s"${_type}${name}蹦蹦跳跳...")
}


object TraitTest {
  def main(args: Array[String]): Unit = {
    val cat = new Cat("Tom")
    cat.roar
    cat.run()
    cat.walk()
    cat.swim()

    val bird = new Bird("Angela")
    bird.fly()
    bird.walk()
    bird.roar
  }
}
------------------------
猫Tom喵喵...
猫Tom疯狂的奔跑...
猫Tom悠闲地漫步...
猫Tom水里游泳...
鸟Angela天空翱翔...
鸟Angela闲庭信步...
鸟Angela叽叽喳喳的叫...
------------------------

6、动态混入

6.1:动态混入的解读

动态混入:未来创建对象时,将特质加进去。

通俗来讲:有些行为同一种动物会有不同表现。我们会在创建对象时,添加相对应的行为。

6.2:动态混入详细案例讲解

//特质trait无法使用构造器,因此用abstract来代替
abstract class Animal(brand:String){
  //抽象属性
  var name:String
  //普通属性
  var _type:String = brand

  //抽象方法
  def roar:Unit
  //普通方法
  def me = s"${_type}:$name"
}

//功能性trait
trait BySwing{
  def fly():Unit // 飞行
}
trait ByFoot{
  def walk():Unit // 走路
  def run():Unit // 跑步
}
trait ByTail{
  def swim():Unit // 游泳
}

// 1、动态强制混入特质:只能定义一个强制混入特质,且必须位于类内首行
// self 是 this 的别名
class Fish extends Animal("鱼"){
  self:ByTail=>						// 强制混入特质语法
  override var name: String = _
  override def roar: Unit = println(s"鱼正在咕噜咕噜吐泡泡") //鱼统一相同行为
}

// 同一个鱼,对于游泳这一行为有不同的表现
object TraitTest {
  def main(args: Array[String]): Unit = {
    // 1、动态强制混入特性  => ByTail行为在创建对象时是强制需要添加的
    // 2、动态非强制混入特质 with,支持多混入 => 其余行为可以自行添加,此处添加了ByFoot行为
    val fish1 = new Fish() with ByTail with ByFoot {
      // 1、动态强制混入特性
      override def swim(): Unit = println(s"${_type}1快速穿梭...")
	  // 2、动态非强制混入特质
      override def walk(): Unit = println(s"${_type}1岸上散步...")
      override def run(): Unit = println(s"${_type}1加速追赶...")
    }
      
    // 3、复合类型 Fish with ByTail【可省略,如:fish1】
    val fish2:Fish with ByTail = new Fish() with ByTail {
      // 1、动态强制混入特性
      override def swim(): Unit = println(s"${_type}2水中漫游...")
    }

    fish1.swim()
    fish1.run()
    fish1.walk()
      
    fish2.swim()
  }
}
------------------1快速穿梭...1加速追赶...1岸上散步...2水中漫游...
------------------

7、抽象类 VS 特质

7.1:抽象类与特质的对比

一般【优先使用特质】:
	1、抽象类在于多类公共属性和行为的抽象,重点在于【封装思想】,本质为类,【单继承,不支持多混入】。
	2、特质(接口)在于一类事物的属性和行为标准定义,重点在于【多态思想,支持多混入】,【动态混入若需		要带参构造.只能使用抽象类】。

7.2:小练习

7.2.1:题目要求
需求说明
	1.现在Book拥有电子版本,可以在多终端上播放
    	属性:作者author:String,书名bookName:String,类型bookType:String
        	内容chapters:Array[String]
        方法:简介resume():Unit
    2.定义Ebook特质
        方法:play()
    3.使Book动态混入Ebook特质,实现play()方法
7.2.2:准备工作——小说文章+书名+作者
val chapters = Array(
      "“春游浩荡,是年年寒食,梨花时节。白锦无纹香烂漫,玉树琼苞堆雪。静夜沉沉,浮光霭霭,冷浸溶溶月。人间天上,烂银霞照通彻。浑似姑射真人,天姿灵秀,意气殊高洁。万蕊参差谁信道,不与群芳同列。浩气清英,仙才卓荦,下土难分别。瑶台归去,洞天方看清绝。”\n\n  作这一首《无俗念》词的,乃南宋末年一位武学名家,有道之士。此人姓丘,名处机,道号长春子,名列全真七子之一,是全真教中出类拔萃的人物。《词品》评论此词道:“长春,世之所谓仙人也,而词之清拔如此”。这首词诵的似是梨花,其实词中真意却是赞誉一位身穿白衣的美貌少女,说她“浑似姑射真人,天姿灵秀,意气殊高洁”,又说她“浩气清英,仙才卓荦”,“不与群芳同列”。词中所颂这美女,乃古墓派传人小龙女。她一生爱穿白衣,当真如风拂玉树,雪裹琼苞,兼之生性清冷,实当得起“冷浸溶溶月”的形容,以“无俗念”三字赠之,可说十分贴切。长春子丘处机和她在终南山上比邻而居,当年一见,便写下这首词来。\n\n  这时丘处机逝世已久,小龙女也已嫁与神雕大侠杨过为妻。在河南少室山山道之上,却另有一个少女,正在低低念诵此词。这少女十**岁年纪,身穿淡黄衣衫,骑着一头青驴,正沿山道缓缓而上,心中默想:“也只有龙姊姊这样的人物,才配得上他。”这一个“他”字,指的自然是神雕大侠杨过了。她也不拉缰绳,任由那青驴信步而行,一路上山。过了良久,她又低声吟道:“欢乐趣,离别苦,就中更有痴儿女。君应有语,渺万里层云,千山暮雪,只影向谁去?”",
      "她腰悬短剑,脸上颇有风尘之色,显是远游已久;韶华如花,正当喜乐无忧之年,可是容色间却隐隐有懊闷意,似是愁思袭人,眉间心上,无计回避。\n\n  这少女姓郭,单名一个襄字,乃大侠郭靖和女侠黄蓉的次女,有个外号叫做“小东邪”。她一驴一剑,只身漫游,原想排遣心中愁闷,岂知酒入愁肠固然愁上加愁,而名山独游,一般的也是愁闷徒增。河南少室山山势颇陡,山道却是一长列宽大的石级,规模宏伟,工程着实不小,那是唐朝高宗为临幸少林寺而开凿,共长八里。郭襄骑着青驴委折而上,只见对面山上五道瀑布飞珠溅玉,奔泻而下,再俯视群山,已如蚁蛭。顺着山道转过一个弯,遥见黄墙碧瓦,好大一座寺院。\n\n  她望着连绵屋宇出了一会神,心想:“少林寺向为天下武学之源,但华山两次论剑,怎地五绝之中并无少林寺高僧?难道寺中和尚自忖没有把握,生怕堕了威名,索性便不去与会?又难道众僧侣修为精湛,名心尽去,武功虽高,却不去和旁人争强赌胜?”她下了青驴,缓步走向寺前,只见树木森森,荫着一片碑林。石碑大半已经毁破,字迹模糊,不知写着些甚么。心想:“便是刻凿在石碑上的字,年深月久之后也须磨灭,如何刻在我心上的,却是时日越久反而越加清晰?”瞥眼只见一块大碑上刻着唐太宗赐少林寺寺僧的御札,嘉许少林寺僧立功平乱。碑文中说唐太宗为秦王时,带兵讨伐王世充,少林寺和尚投军立功,最著者共一十三人。其中只昙宗一僧受封为大将军,其余十二僧不愿为官,唐太宗各赐紫罗袈裟一袭。她神驰想象:“当隋唐之际,少林寺武功便已名驰天下,数百年来精益求精,这寺中卧虎藏龙,不知有多少好手。”郭襄自和杨过、小龙女夫妇在华山绝顶分手后,三年来没得到他二人半点音讯。她心中长自记挂,于是禀明父母,说要出来游山玩水,实则是打听杨过的消息,她倒也不一定要和他夫妇会面,只须听到一些杨过如何在江湖上行侠的讯息,也便心满意足了。偏生一别之后,他夫妇从此便不在江湖上露面,不知到了何处隐居,郭襄自北而南,又从东至西,几乎踏遍了大半个中原,始终没听到有人说起神雕大侠杨过的近讯。这一日她到了河南,想起少林寺中有一位僧人无色禅师是杨过的好友,自己十六岁生日之时,无色瞧在杨过的面上,曾托人送来一件礼物,虽然从未和他见过面,但不妨去问他一问,说不定他会知道杨过的踪迹,这才上少林寺来。正出神间,忽听得碑林旁树丛后传出一阵铁链当啷之声,一人诵念佛经:“是时药叉共王立要,即于无量百千万亿大众之中,说胜妙伽他曰:由爱故生忧,由爱故生怖;若离于爱者,无忧亦无怖……”郭襄听了这四句偈言,不由得痴了,心中默默念道:“由爱故生忧,由爱故生怖;若离于爱者,无忧亦无怖。”只听得铁链拖地和念佛之声渐渐远去。郭襄低声道:“我要问他,如何才能离于爱,如何能无忧无怖?”随手将驴缰在树上一绕,拨开树丛,追了过去。只见树后是一条上山的小径,一个僧人挑了一对大桶,正缓缓往山上走去。郭襄快步跟上,奔到距那僧人七八丈处,不由得吃了一惊,只见那僧人挑的是一对大铁桶,比之寻常水桶大了两倍有余,那僧人颈中、手上、脚上,更绕满了粗大的铁链,行走时铁链拖地,不停发出声响。这对大铁桶本身只怕便有二百来斤,桶中装满了水,重量更是惊人。郭襄叫道:“大和尚,请留步,小女子有句话请教。”",
      "那僧人回过头来,两人相对,都是一愕。原来这僧人便是觉远,三年以前,两人在华山绝顶曾有一面之缘。郭襄知他虽然生性迂腐,但内功深湛,不在当世任何高手之下,便道:“我道是谁,原来是觉远大师。你如何变成了这等模样?”觉远点了点头,微微一笑,合十行礼,并不答话,转身便走。郭襄叫道:“觉远大师,你不认得我了么?我是郭襄啊。”觉远又是回首一笑,点了点头,这次更不停步。郭襄又道:“是谁用铁链绑住了你?如何这般虐待你?”觉远左掌伸到脑后摇了几摇,示意她不必再问。\n\n  郭襄见了这等怪事,如何肯不弄个明白?当下飞步追赶,想抢在他面前拦住,岂知觉远虽然全身带了铁链,又挑着一对大铁桶,但郭襄快步追赶,始终抢不到他身前。郭襄童心大起,展开家传轻功,双足一点,身子飞起,伸手往铁桶边上抓去,眼见这一下必能抓中。不料落手时终究还是差了两寸。郭襄叫道:“大和尚,这般好本事,我非追上你不可。”但见觉远不疾不徐的迈步而行,铁链声当啷当啷有如乐音,越走越高,直至后山。郭襄直奔得气喘渐急,但仍和他相距丈余,不由得心中佩服:“爹爹妈妈在华山之上,便说这位大和尚武功极高,当时我还不大相信,今日一试,才知爹妈的话果然不错。”只见觉远转身走到一间小屋之后,将铁桶中的两桶水都倒进了一口井中。郭襄大奇,叫道:“大和尚,你莫非疯了,挑水倒在井中干么?”觉远神色平和,只摇了摇头。郭襄忽有所悟,笑道:“啊,你是在练一门高深的武功。”觉远又摇了摇头。郭襄心中着恼,说道:“我刚才明明听得你在念经,又不是哑了,怎地不答我的话?”觉远合十行礼,脸上似有歉意,一言不发,挑了铁桶便下山去。郭襄探头井口向下望去,只见井水清澈,也无特异之处,怔怔望着觉远的背影,心中满是疑窦。她适才一阵追赶,微感心浮气躁,于是坐在井栏圈上,观看四下风景,这时置身处已高于少林寺所有屋宇,但见少室山层崖刺天,横若列屏,崖下风烟飘渺,寺中钟声随风送上,令人一洗烦俗之气。郭襄心想:“这和尚的弟子不知在哪里,和尚既不肯说,我去问那个少年便了。”当下信步落山,想去找觉远的弟子张君宝来问。走了一程,忽听得铁链声响,觉远又挑了水上来。郭襄闪身躲在树后,心想:“我暗中瞧瞧他到底在捣甚么鬼。”铁链声渐近,只见觉远仍是挑着那对铁桶,手中却拿着一本书,全神贯注的轻声诵读。郭襄待他走到身边,猛地里跃出,叫道:“大和尚,你看甚么书?”\n\n  觉远失声叫道:“啊哟,吓了我一跳,原来是你。”郭襄笑道:“你装哑巴装不成了罢,怎么说话了?”觉远微有惊色,向左右一望,摇了摇手。郭襄道:“你怕甚么?”觉远还未回答,突然树林中转出两个灰衣僧人,一高一矮。那瘦长僧人喝道:“觉远,不守戒法,擅自开口说话,何况又和庙外生人对答,更何况又和年轻女子说话?这便见戒律堂首座去。”觉远垂头丧气,点了点头,跟在那两个僧人之后。郭襄大为惊怒,喝道:“天下还有不许人说话的规矩么?我识得这位大师,我自跟他说话,干你们何事?”那瘦长僧人白眼一翻,说道:“千年以来,少林寺向不许女流擅入。姑娘请下山去罢,免得自讨没趣。”郭襄心中更怒,说道:“女流便怎样?难道女子便不是人?你们干么难为这位觉远大师?既用铁链捆绑他,又不许他说话?”那僧人冷冷的道:“本寺之事,便是皇帝也管不着。何劳姑娘多问?”",
      "郭襄怒道:“这位大师是忠厚老实的好人,你们欺他仁善,便这般折磨于他,哼哼,天鸣禅师呢?无色和尚、无相和尚在哪里?你去叫他们出来,我倒要问问这个道理。”两个僧人听了都是一惊。天鸣禅师是少林寺方丈,无色禅师是本寺罗汉堂首座,无相禅师是达摩堂首座,三人位望尊崇,寺中僧侣向来只称“老方丈”、“罗汉堂座师”、“达摩堂座师”,从来不敢提及法名,岂知一个年轻女子竟敢上山来大呼小叫,直斥其名。那两名僧人都是戒律堂首座的弟子,奉了座师之命,监视觉远,这时听郭襄言语莽撞,那瘦长僧人喝道:“女施主再在佛门清净之地滋扰,莫怪小僧无礼。”\n\n  郭襄道:“难道我还怕了你这和尚?你快快把觉远大师身上的铁链除去,那便算了,否则我找天鸣老和尚算帐去。”那矮僧听郭襄出言无状,又见她腰悬短剑,沉着嗓子道:“你把兵刃留下,我们也不来跟你一般见识,快下山去罢。”郭襄摘下短剑,双手托起,冷笑道:“好罢,谨遵台命。”那矮僧自幼在少林寺出家,一向听师伯、师叔、师兄们说少林寺是天下武学的总源,又听说不论名望多大、本领多强的武林高手,从不敢携带兵刃走进少林寺出门。这年轻姑娘虽然未入寺门,但已在少林寺范围之内,只道她真是怕了,乖乖交出短剑,于是伸手便去接剑。他手指刚碰到剑鞘,突然间手臂剧震,如中电掣,但觉一股强力从短剑上传了过来,推得他向后急仰,立足不定,登时摔倒。他身在斜坡之上,一经摔倒,便骨碌碌的向下滚了数丈,好容易硬生生的撑住,这才不再滚动。那瘦长僧人又惊又怒,喝道:“你吃了狮子心豹子胆,竟到少林寺撒野来啦!”转过身来,踏上一步,右手一拳击出,左掌跟着在右拳上一搭,变成双掌下劈,正是“闯少林”第二十八势“翻身劈击”。郭襄握住剑柄,连剑带鞘向他肩头砸去。那僧人沉肩回掌,来抓剑鞘。觉远在旁瞧得惶急,大叫:“别动手,别动手!有话好说。”便在此时,那僧人右手已抓住剑鞘,正却运劲里夺,猛觉手心一震,双臂隐隐酸麻,只叫得一声:“不好!”郭襄左腿横扫,已将他踢下坡去。他所受的这一招比那矮僧重得多,一路翻滚,头脸上擦出不少鲜血,这才停住。郭襄心道:“我上少林寺来是打听大哥哥的讯息,平白无端的跟他们动手,当真好没来由。”眼见觉远愁眉苦脸的站在一旁,当即抽出短剑,便往他手脚上的铁链削去。这短剑虽非稀世奇珍,却也是极锋锐的利器,只听得当啷啷几声响,铁链断了三条。觉远连呼:“使不得,使不得!”郭襄道:“甚么使不得?”指着正向寺内奔去的高矮二僧说道:“这两个恶和尚定是奔去报讯,咱们快走。你那个姓张的小徒儿呢?带了他一起走罢!”觉远只是摇手。忽听得身后一人说道:“多谢姑娘关怀,小的在这儿。”\n\n  郭襄回过头来,只见身后站着个十六七岁的少年,粗眉大眼,身材魁伟,脸上却犹带稚气,正是三年前曾在华山之巅会过的张君宝。比之当日,他身形已高了许多,但容貌无甚改变。郭襄大喜,说道:“这里的恶和尚欺侮你师父,咱们走罢。”张君宝摇头道:“没有谁欺侮我师父啊。”郭襄指着觉远道:“那两个恶和尚用铁链锁着你师父,连一句话也不许他说,还不是欺侮?”觉远苦笑摇头,指了指山下,示意郭襄及早脱身,免惹事端。郭襄明知少林寺中武功胜过她的人不计其数,但既见了眼前的不平之事,决不能便此撒手不顾;可是却又担心寺中好手出来截拦,当下一手拉了觉远,一手拉了张君宝,顿足道:“快走快走,有甚么事,下山去慢慢说不好么?”两人只是不动。忽见山坡下寺院边门中冲出七八名僧人,手提齐眉木棍,吆喝道:“哪里来的野姑娘,胆敢来少林寺撒野?”张君宝提起嗓子叫道:“各位师兄不得无礼,这位是……”郭襄忙道:“别说我名字。”她想今日的祸事看来闯得不小,说不定闹下去会不可收拾,可别牵累到爹爹妈妈,又补上一句:“咱们翻山走罢!千万别提我爹爹妈妈和朋友的姓名。”只听得背后山顶上吆喝声响,又涌出七八名僧人来。郭襄见前后都出现了僧人,秀眉深蹙,急道:“你们两个婆婆妈妈,没点男子汉气概!到底走不走?”张君宝道:“师父,郭姑娘一片好意……”",
      "便在此时,下面边门中又窜出四名黄衣僧人,飕飕飕的奔上坡来,手中都没兵器,但身法迅捷,衣襟带风,武功颇为了得。郭襄见这般情势,便想单独脱身亦已不能,索性凝气卓立,静观其变。当先一名僧人奔到离她四丈之处,朗声说道:“罗汉堂首座尊师传谕:着来人放下兵刃,在山下一苇亭中陈明详情,听由法谕。”\n\n  郭襄冷笑道:“少林寺的大和尚官派十足,官腔打得倒好听。请问各位大和尚做的是大宋皇帝的官儿呢,还是做蒙古皇帝的官?”这时淮水以北,大宋国土均已沦陷,少林寺所在之地自也早该归蒙古管,只是蒙古大军连年进攻襄阳不克,忙于调兵遣将,也无余力来理会丛林寺观的事,因此少林寺一如其旧,与前并无不同。那僧人听郭襄讥刺之言甚是厉害,不由得脸上一红,心中也觉对外人下令传谕有些不妥,合十说道:“不知女施主何事光临敝寺,且请放下兵刃,赴山下一苇亭中奉茶说话。”郭襄听他语转和缓,便想乘此收蓬,说道:“你们不让我进寺,我便希罕了?哼,难道少林寺中有宝,我见一见便沾了光么?”向张君宝使个眼色,低声道:“到底走不走?”张君宝摇摇头,嘴角向觉远一努,意思说是要服侍师父。郭襄朗声道:“好,那我不管啦,我走了。”拔步便下坡去。第一名黄衣僧侧身让开。第二名和第三名黄衣僧却同时伸手一拦,齐声道:“且慢,放下了兵刃。”郭襄眉毛一扬,手按剑柄。第一名僧人道:“我们也不敢留着女施主的兵刃。女施主一到山下,我们立即将宝剑送上,这是少林寺千年来的规矩,还请包涵。”郭襄听他言语有礼,心下踌躇:“倘若不留短剑,势必有场争斗,我孤身一人,如何是阖寺僧众的敌手?但若留下短剑,岂不将外公、爹爹、妈妈、大哥哥、龙姊姊的面子一古脑儿都丢得干净?”她一时沉吟未决,蓦地里眼前黄影晃动,一人喝道:“到少林寺来既带剑,又伤人,世上焉有是理?”跟着劲风飒然,五只手指往剑鞘上抓下来。这僧人若不贸然出手,郭襄一番迟疑之后,多半便会将短剑留下。她和乃姊郭芙的性子大不相同,虽然豪爽,却不鲁莽,眼前处境既极度不利,便会暂忍一时之气,日后再去和外公、爹妈商量,回头找这场子,但对方突然逞强,岂能眼睁睁的让他将剑夺去?那僧人的擒拿手法既狠且巧,一抓住剑鞘,心想郭襄定会向里回夺,一个和尚跟一个年轻女子拉拉扯扯,大是不雅,当下运劲向左斜推,跟着抓而向右。郭襄被他这么一推一抓,果然已拿不牢剑鞘,当即握住剑柄,刷的一声,寒光出匣。那僧人右手将剑鞘夺了过去,左手却有两根手指被短剑顺势割断,剧痛之下,抛下剑鞘,往旁退开。\n\n  众僧人见同门受伤,无不惊怒,挥杖舞棍,一齐攻来。郭襄心想:“一不做二不休,反正今日已不能善罢。”当下使出家传的“落英剑法”,便往山下冲去。众僧人排成三列,仰面挡住。那“落英剑法”乃黄药师从“落英掌法”的路子中演化来,虽不若“玉箫剑法”的精妙,却也是桃花岛的一绝,但见青光激荡,剑花点点,便似落英缤纷,四散而下,霎时间僧人中又有两人受伤。但背后数名僧人跟着抢到,居高临下的夹攻。按理郭襄早已抵挡不住,只是少林僧众慈悲为本,不愿伤她性命,所出招数都非杀手,只求将她打倒,训诫一番,扣下兵刃,将她逐下山去。可是郭襄剑光错落,却也不易攻近身去。众僧初时只道一个妙龄女郎,还不轻易打发?待见她剑法精奇,始知她若非名门之女,便是名师之徒,多半得罪不得,出招时更有分寸,一面急报罗汉堂首座无色禅师。正斗之间,一个身材高瘦老年僧人缓步走近,双手笼在袖中,微笑观斗。两名僧人走到他身前,低声禀告了几句。郭襄已斗得气喘吁吁,剑法凌乱,大声喝道:“说甚么天下武学之源,原来是十多个和尚一拥而上,倚多为胜。”那老僧便是罗汉堂首座无色禅师,听她这么说,便道:“各人住手!”众僧人立时罢手跃开。无色禅师道:“姑娘贵姓,令尊和令师是谁?光临少林寺,不知有何贵干?”郭襄心道:“我爹娘的姓名不能告诉你。我到少林寺来是为了打听大哥哥的讯息,那也不能当众述说。眼下已闹成这等模样,日后爹娘和大哥哥知道了定要怪我,不如悄悄的溜了罢。”说道:“我的姓名不能跟你说,我不过见山上风景优美,这便上来游览玩耍。原来少林寺比皇宫内院还要厉害,动不动便要扣人家兵刃。请问大师,我进了贵寺的山门没有?当日达摩祖师传下武艺,想来也不过教众僧侣强身健体,便于精进修为,想不到少林寺名头越大,武功越高,恃众逞强的名头也越来越响。好,你们要扣我兵刃,这便留下,除非将我杀了,否则今日之事江湖上不会无人知晓。”她本来伶牙俐齿,这件事也并非全是她的过错,一席话只将无色禅师说得哑口无言。郭襄鉴貌辨色,心想:“这番胡闹我固怕人知晓,看来少林寺更加不愿张扬。十多个和尚围斗一个年轻姑娘,说出去有甚么好听?”当下哼的一声,将短剑往地下一掷,举步便行。"
)
val book = EBook("倚天屠龙记","张天爱","武侠小说")
7.2.3:具体展示

EBook

package scala.cha06

import java.util.regex.{Matcher, Pattern}
import scala.collection.mutable.ArrayBuffer

class Book(title:String,author:Array[String]) {
   val _title:String = title
   val _authors:Array[String] = author

  //辅助构造器:实现类型转换
  def this(title:String,author:String) = this(title,Array(author))

  // 设计业务:解决问题前置化(在源头解决问题)
  //1.判断书名是否包含“xxx”的方法
  def titleContains(sub:String) = _title.contains(sub)
  //2.判断作者名是否以“xxx”打头的方法
  def authorContains(name:String) = { _authors.count(_.startsWith(name))>0 }

  def getTitle = _title

  override def toString: String = s"$title\t${_authors.mkString(",")}"
}

object Book{
  //主构造器
  def apply(title: String, author: Array[String]): Book = new Book(title, author)
  //辅助构造器
  def apply(title: String, author: String): Book = new Book(title, author)
}

//=================================================
trait EAction{
  //看某一个章节
  def play(chapterNo:Int):Unit
  //判断是否继续观看并展示章节内容
  def play()
}

class EBook(title:String,author:Array[String],bookType:String)
  extends Book(title,author) with EAction {
  val _bookType:String = bookType //书类型
  val _chapters:ArrayBuffer[String] = ArrayBuffer() //章节数

  def this(title:String,author:String,bookType:String) = this(title,Array(author),bookType)

  def chapterCount = _chapters.size
  def resume:Unit = println(s"类型:${_bookType} ,标题:${_title},作者:${_authors.mkString("、")},章节数:${_chapters.size}")
  //将章节加进来
  def addChapter(chapter:String):Unit = {
    _chapters.append(chapter)
  }
      
  //输出文章内容(本质是java,可在java中运行验证再复制于此)
  private def showChapter(chapter:String): Unit ={
    val patLine: Pattern = Pattern.compile("([^\n]{1,30})")//一行30个
    val matcher: Matcher = patLine.matcher(chapter)
    while (matcher.find){
      System.out.println(matcher.group(1))
    }
  }

  override def play(chapterNo: Int): Unit = {
    if(chapterNo>=1 && chapterNo<=_chapters.size){
      showChapter(_chapters(chapterNo-1))
    }else{
      println("无此章节")
    }
  }

  override def play(): Unit = {
    import scala.util.control.Breaks._ //控制
    import scala.io.StdIn._ //控制台输入
    //给个选择,跳出循环
    breakable({
      var chapterNo = 1 //当前读取的章节数
      while (chapterNo<=chapterCount){
        play(chapterNo) //跳转至第chapterNo章节,展示当前章节内容
        print("是否继续读取:")
        //readLine():读取控制台输入内容,用于判断是否继续读取
        if(!readLine().matches("y|Y")){
          break() //跳出breakable结构
        }
        chapterNo += 1
      }
    })
  }
}

object EBook{
  //简化构造方式
  def apply(title: String, author: String, bookType: String): EBook = new EBook(title, author, bookType)
}

测试结果

package scala.cha06

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    val chapters = Array(
      "“春游浩荡,是年年寒食,梨花时节。白锦无纹香烂漫,玉树琼苞堆雪。静夜沉沉,浮光霭霭,冷浸溶溶月。人间天上,烂银霞照通彻。浑似姑射真人,天姿灵秀,意气殊高洁。万蕊参差谁信道,不与群芳同列。浩气清英,仙才卓荦,下土难分别。瑶台归去,洞天方看清绝。”\n\n  作这一首《无俗念》词的,乃南宋末年一位武学名家,有道之士。此人姓丘,名处机,道号长春子,名列全真七子之一,是全真教中出类拔萃的人物。《词品》评论此词道:“长春,世之所谓仙人也,而词之清拔如此”。这首词诵的似是梨花,其实词中真意却是赞誉一位身穿白衣的美貌少女,说她“浑似姑射真人,天姿灵秀,意气殊高洁”,又说她“浩气清英,仙才卓荦”,“不与群芳同列”。词中所颂这美女,乃古墓派传人小龙女。她一生爱穿白衣,当真如风拂玉树,雪裹琼苞,兼之生性清冷,实当得起“冷浸溶溶月”的形容,以“无俗念”三字赠之,可说十分贴切。长春子丘处机和她在终南山上比邻而居,当年一见,便写下这首词来。\n\n  这时丘处机逝世已久,小龙女也已嫁与神雕大侠杨过为妻。在河南少室山山道之上,却另有一个少女,正在低低念诵此词。这少女十**岁年纪,身穿淡黄衣衫,骑着一头青驴,正沿山道缓缓而上,心中默想:“也只有龙姊姊这样的人物,才配得上他。”这一个“他”字,指的自然是神雕大侠杨过了。她也不拉缰绳,任由那青驴信步而行,一路上山。过了良久,她又低声吟道:“欢乐趣,离别苦,就中更有痴儿女。君应有语,渺万里层云,千山暮雪,只影向谁去?”",
      "她腰悬短剑,脸上颇有风尘之色,显是远游已久;韶华如花,正当喜乐无忧之年,可是容色间却隐隐有懊闷意,似是愁思袭人,眉间心上,无计回避。\n\n  这少女姓郭,单名一个襄字,乃大侠郭靖和女侠黄蓉的次女,有个外号叫做“小东邪”。她一驴一剑,只身漫游,原想排遣心中愁闷,岂知酒入愁肠固然愁上加愁,而名山独游,一般的也是愁闷徒增。河南少室山山势颇陡,山道却是一长列宽大的石级,规模宏伟,工程着实不小,那是唐朝高宗为临幸少林寺而开凿,共长八里。郭襄骑着青驴委折而上,只见对面山上五道瀑布飞珠溅玉,奔泻而下,再俯视群山,已如蚁蛭。顺着山道转过一个弯,遥见黄墙碧瓦,好大一座寺院。\n\n  她望着连绵屋宇出了一会神,心想:“少林寺向为天下武学之源,但华山两次论剑,怎地五绝之中并无少林寺高僧?难道寺中和尚自忖没有把握,生怕堕了威名,索性便不去与会?又难道众僧侣修为精湛,名心尽去,武功虽高,却不去和旁人争强赌胜?”她下了青驴,缓步走向寺前,只见树木森森,荫着一片碑林。石碑大半已经毁破,字迹模糊,不知写着些甚么。心想:“便是刻凿在石碑上的字,年深月久之后也须磨灭,如何刻在我心上的,却是时日越久反而越加清晰?”瞥眼只见一块大碑上刻着唐太宗赐少林寺寺僧的御札,嘉许少林寺僧立功平乱。碑文中说唐太宗为秦王时,带兵讨伐王世充,少林寺和尚投军立功,最著者共一十三人。其中只昙宗一僧受封为大将军,其余十二僧不愿为官,唐太宗各赐紫罗袈裟一袭。她神驰想象:“当隋唐之际,少林寺武功便已名驰天下,数百年来精益求精,这寺中卧虎藏龙,不知有多少好手。”郭襄自和杨过、小龙女夫妇在华山绝顶分手后,三年来没得到他二人半点音讯。她心中长自记挂,于是禀明父母,说要出来游山玩水,实则是打听杨过的消息,她倒也不一定要和他夫妇会面,只须听到一些杨过如何在江湖上行侠的讯息,也便心满意足了。偏生一别之后,他夫妇从此便不在江湖上露面,不知到了何处隐居,郭襄自北而南,又从东至西,几乎踏遍了大半个中原,始终没听到有人说起神雕大侠杨过的近讯。这一日她到了河南,想起少林寺中有一位僧人无色禅师是杨过的好友,自己十六岁生日之时,无色瞧在杨过的面上,曾托人送来一件礼物,虽然从未和他见过面,但不妨去问他一问,说不定他会知道杨过的踪迹,这才上少林寺来。正出神间,忽听得碑林旁树丛后传出一阵铁链当啷之声,一人诵念佛经:“是时药叉共王立要,即于无量百千万亿大众之中,说胜妙伽他曰:由爱故生忧,由爱故生怖;若离于爱者,无忧亦无怖……”郭襄听了这四句偈言,不由得痴了,心中默默念道:“由爱故生忧,由爱故生怖;若离于爱者,无忧亦无怖。”只听得铁链拖地和念佛之声渐渐远去。郭襄低声道:“我要问他,如何才能离于爱,如何能无忧无怖?”随手将驴缰在树上一绕,拨开树丛,追了过去。只见树后是一条上山的小径,一个僧人挑了一对大桶,正缓缓往山上走去。郭襄快步跟上,奔到距那僧人七八丈处,不由得吃了一惊,只见那僧人挑的是一对大铁桶,比之寻常水桶大了两倍有余,那僧人颈中、手上、脚上,更绕满了粗大的铁链,行走时铁链拖地,不停发出声响。这对大铁桶本身只怕便有二百来斤,桶中装满了水,重量更是惊人。郭襄叫道:“大和尚,请留步,小女子有句话请教。”",
      "那僧人回过头来,两人相对,都是一愕。原来这僧人便是觉远,三年以前,两人在华山绝顶曾有一面之缘。郭襄知他虽然生性迂腐,但内功深湛,不在当世任何高手之下,便道:“我道是谁,原来是觉远大师。你如何变成了这等模样?”觉远点了点头,微微一笑,合十行礼,并不答话,转身便走。郭襄叫道:“觉远大师,你不认得我了么?我是郭襄啊。”觉远又是回首一笑,点了点头,这次更不停步。郭襄又道:“是谁用铁链绑住了你?如何这般虐待你?”觉远左掌伸到脑后摇了几摇,示意她不必再问。\n\n  郭襄见了这等怪事,如何肯不弄个明白?当下飞步追赶,想抢在他面前拦住,岂知觉远虽然全身带了铁链,又挑着一对大铁桶,但郭襄快步追赶,始终抢不到他身前。郭襄童心大起,展开家传轻功,双足一点,身子飞起,伸手往铁桶边上抓去,眼见这一下必能抓中。不料落手时终究还是差了两寸。郭襄叫道:“大和尚,这般好本事,我非追上你不可。”但见觉远不疾不徐的迈步而行,铁链声当啷当啷有如乐音,越走越高,直至后山。郭襄直奔得气喘渐急,但仍和他相距丈余,不由得心中佩服:“爹爹妈妈在华山之上,便说这位大和尚武功极高,当时我还不大相信,今日一试,才知爹妈的话果然不错。”只见觉远转身走到一间小屋之后,将铁桶中的两桶水都倒进了一口井中。郭襄大奇,叫道:“大和尚,你莫非疯了,挑水倒在井中干么?”觉远神色平和,只摇了摇头。郭襄忽有所悟,笑道:“啊,你是在练一门高深的武功。”觉远又摇了摇头。郭襄心中着恼,说道:“我刚才明明听得你在念经,又不是哑了,怎地不答我的话?”觉远合十行礼,脸上似有歉意,一言不发,挑了铁桶便下山去。郭襄探头井口向下望去,只见井水清澈,也无特异之处,怔怔望着觉远的背影,心中满是疑窦。她适才一阵追赶,微感心浮气躁,于是坐在井栏圈上,观看四下风景,这时置身处已高于少林寺所有屋宇,但见少室山层崖刺天,横若列屏,崖下风烟飘渺,寺中钟声随风送上,令人一洗烦俗之气。郭襄心想:“这和尚的弟子不知在哪里,和尚既不肯说,我去问那个少年便了。”当下信步落山,想去找觉远的弟子张君宝来问。走了一程,忽听得铁链声响,觉远又挑了水上来。郭襄闪身躲在树后,心想:“我暗中瞧瞧他到底在捣甚么鬼。”铁链声渐近,只见觉远仍是挑着那对铁桶,手中却拿着一本书,全神贯注的轻声诵读。郭襄待他走到身边,猛地里跃出,叫道:“大和尚,你看甚么书?”\n\n  觉远失声叫道:“啊哟,吓了我一跳,原来是你。”郭襄笑道:“你装哑巴装不成了罢,怎么说话了?”觉远微有惊色,向左右一望,摇了摇手。郭襄道:“你怕甚么?”觉远还未回答,突然树林中转出两个灰衣僧人,一高一矮。那瘦长僧人喝道:“觉远,不守戒法,擅自开口说话,何况又和庙外生人对答,更何况又和年轻女子说话?这便见戒律堂首座去。”觉远垂头丧气,点了点头,跟在那两个僧人之后。郭襄大为惊怒,喝道:“天下还有不许人说话的规矩么?我识得这位大师,我自跟他说话,干你们何事?”那瘦长僧人白眼一翻,说道:“千年以来,少林寺向不许女流擅入。姑娘请下山去罢,免得自讨没趣。”郭襄心中更怒,说道:“女流便怎样?难道女子便不是人?你们干么难为这位觉远大师?既用铁链捆绑他,又不许他说话?”那僧人冷冷的道:“本寺之事,便是皇帝也管不着。何劳姑娘多问?”",
      "郭襄怒道:“这位大师是忠厚老实的好人,你们欺他仁善,便这般折磨于他,哼哼,天鸣禅师呢?无色和尚、无相和尚在哪里?你去叫他们出来,我倒要问问这个道理。”两个僧人听了都是一惊。天鸣禅师是少林寺方丈,无色禅师是本寺罗汉堂首座,无相禅师是达摩堂首座,三人位望尊崇,寺中僧侣向来只称“老方丈”、“罗汉堂座师”、“达摩堂座师”,从来不敢提及法名,岂知一个年轻女子竟敢上山来大呼小叫,直斥其名。那两名僧人都是戒律堂首座的弟子,奉了座师之命,监视觉远,这时听郭襄言语莽撞,那瘦长僧人喝道:“女施主再在佛门清净之地滋扰,莫怪小僧无礼。”\n\n  郭襄道:“难道我还怕了你这和尚?你快快把觉远大师身上的铁链除去,那便算了,否则我找天鸣老和尚算帐去。”那矮僧听郭襄出言无状,又见她腰悬短剑,沉着嗓子道:“你把兵刃留下,我们也不来跟你一般见识,快下山去罢。”郭襄摘下短剑,双手托起,冷笑道:“好罢,谨遵台命。”那矮僧自幼在少林寺出家,一向听师伯、师叔、师兄们说少林寺是天下武学的总源,又听说不论名望多大、本领多强的武林高手,从不敢携带兵刃走进少林寺出门。这年轻姑娘虽然未入寺门,但已在少林寺范围之内,只道她真是怕了,乖乖交出短剑,于是伸手便去接剑。他手指刚碰到剑鞘,突然间手臂剧震,如中电掣,但觉一股强力从短剑上传了过来,推得他向后急仰,立足不定,登时摔倒。他身在斜坡之上,一经摔倒,便骨碌碌的向下滚了数丈,好容易硬生生的撑住,这才不再滚动。那瘦长僧人又惊又怒,喝道:“你吃了狮子心豹子胆,竟到少林寺撒野来啦!”转过身来,踏上一步,右手一拳击出,左掌跟着在右拳上一搭,变成双掌下劈,正是“闯少林”第二十八势“翻身劈击”。郭襄握住剑柄,连剑带鞘向他肩头砸去。那僧人沉肩回掌,来抓剑鞘。觉远在旁瞧得惶急,大叫:“别动手,别动手!有话好说。”便在此时,那僧人右手已抓住剑鞘,正却运劲里夺,猛觉手心一震,双臂隐隐酸麻,只叫得一声:“不好!”郭襄左腿横扫,已将他踢下坡去。他所受的这一招比那矮僧重得多,一路翻滚,头脸上擦出不少鲜血,这才停住。郭襄心道:“我上少林寺来是打听大哥哥的讯息,平白无端的跟他们动手,当真好没来由。”眼见觉远愁眉苦脸的站在一旁,当即抽出短剑,便往他手脚上的铁链削去。这短剑虽非稀世奇珍,却也是极锋锐的利器,只听得当啷啷几声响,铁链断了三条。觉远连呼:“使不得,使不得!”郭襄道:“甚么使不得?”指着正向寺内奔去的高矮二僧说道:“这两个恶和尚定是奔去报讯,咱们快走。你那个姓张的小徒儿呢?带了他一起走罢!”觉远只是摇手。忽听得身后一人说道:“多谢姑娘关怀,小的在这儿。”\n\n  郭襄回过头来,只见身后站着个十六七岁的少年,粗眉大眼,身材魁伟,脸上却犹带稚气,正是三年前曾在华山之巅会过的张君宝。比之当日,他身形已高了许多,但容貌无甚改变。郭襄大喜,说道:“这里的恶和尚欺侮你师父,咱们走罢。”张君宝摇头道:“没有谁欺侮我师父啊。”郭襄指着觉远道:“那两个恶和尚用铁链锁着你师父,连一句话也不许他说,还不是欺侮?”觉远苦笑摇头,指了指山下,示意郭襄及早脱身,免惹事端。郭襄明知少林寺中武功胜过她的人不计其数,但既见了眼前的不平之事,决不能便此撒手不顾;可是却又担心寺中好手出来截拦,当下一手拉了觉远,一手拉了张君宝,顿足道:“快走快走,有甚么事,下山去慢慢说不好么?”两人只是不动。忽见山坡下寺院边门中冲出七八名僧人,手提齐眉木棍,吆喝道:“哪里来的野姑娘,胆敢来少林寺撒野?”张君宝提起嗓子叫道:“各位师兄不得无礼,这位是……”郭襄忙道:“别说我名字。”她想今日的祸事看来闯得不小,说不定闹下去会不可收拾,可别牵累到爹爹妈妈,又补上一句:“咱们翻山走罢!千万别提我爹爹妈妈和朋友的姓名。”只听得背后山顶上吆喝声响,又涌出七八名僧人来。郭襄见前后都出现了僧人,秀眉深蹙,急道:“你们两个婆婆妈妈,没点男子汉气概!到底走不走?”张君宝道:“师父,郭姑娘一片好意……”",
      "便在此时,下面边门中又窜出四名黄衣僧人,飕飕飕的奔上坡来,手中都没兵器,但身法迅捷,衣襟带风,武功颇为了得。郭襄见这般情势,便想单独脱身亦已不能,索性凝气卓立,静观其变。当先一名僧人奔到离她四丈之处,朗声说道:“罗汉堂首座尊师传谕:着来人放下兵刃,在山下一苇亭中陈明详情,听由法谕。”\n\n  郭襄冷笑道:“少林寺的大和尚官派十足,官腔打得倒好听。请问各位大和尚做的是大宋皇帝的官儿呢,还是做蒙古皇帝的官?”这时淮水以北,大宋国土均已沦陷,少林寺所在之地自也早该归蒙古管,只是蒙古大军连年进攻襄阳不克,忙于调兵遣将,也无余力来理会丛林寺观的事,因此少林寺一如其旧,与前并无不同。那僧人听郭襄讥刺之言甚是厉害,不由得脸上一红,心中也觉对外人下令传谕有些不妥,合十说道:“不知女施主何事光临敝寺,且请放下兵刃,赴山下一苇亭中奉茶说话。”郭襄听他语转和缓,便想乘此收蓬,说道:“你们不让我进寺,我便希罕了?哼,难道少林寺中有宝,我见一见便沾了光么?”向张君宝使个眼色,低声道:“到底走不走?”张君宝摇摇头,嘴角向觉远一努,意思说是要服侍师父。郭襄朗声道:“好,那我不管啦,我走了。”拔步便下坡去。第一名黄衣僧侧身让开。第二名和第三名黄衣僧却同时伸手一拦,齐声道:“且慢,放下了兵刃。”郭襄眉毛一扬,手按剑柄。第一名僧人道:“我们也不敢留着女施主的兵刃。女施主一到山下,我们立即将宝剑送上,这是少林寺千年来的规矩,还请包涵。”郭襄听他言语有礼,心下踌躇:“倘若不留短剑,势必有场争斗,我孤身一人,如何是阖寺僧众的敌手?但若留下短剑,岂不将外公、爹爹、妈妈、大哥哥、龙姊姊的面子一古脑儿都丢得干净?”她一时沉吟未决,蓦地里眼前黄影晃动,一人喝道:“到少林寺来既带剑,又伤人,世上焉有是理?”跟着劲风飒然,五只手指往剑鞘上抓下来。这僧人若不贸然出手,郭襄一番迟疑之后,多半便会将短剑留下。她和乃姊郭芙的性子大不相同,虽然豪爽,却不鲁莽,眼前处境既极度不利,便会暂忍一时之气,日后再去和外公、爹妈商量,回头找这场子,但对方突然逞强,岂能眼睁睁的让他将剑夺去?那僧人的擒拿手法既狠且巧,一抓住剑鞘,心想郭襄定会向里回夺,一个和尚跟一个年轻女子拉拉扯扯,大是不雅,当下运劲向左斜推,跟着抓而向右。郭襄被他这么一推一抓,果然已拿不牢剑鞘,当即握住剑柄,刷的一声,寒光出匣。那僧人右手将剑鞘夺了过去,左手却有两根手指被短剑顺势割断,剧痛之下,抛下剑鞘,往旁退开。\n\n  众僧人见同门受伤,无不惊怒,挥杖舞棍,一齐攻来。郭襄心想:“一不做二不休,反正今日已不能善罢。”当下使出家传的“落英剑法”,便往山下冲去。众僧人排成三列,仰面挡住。那“落英剑法”乃黄药师从“落英掌法”的路子中演化来,虽不若“玉箫剑法”的精妙,却也是桃花岛的一绝,但见青光激荡,剑花点点,便似落英缤纷,四散而下,霎时间僧人中又有两人受伤。但背后数名僧人跟着抢到,居高临下的夹攻。按理郭襄早已抵挡不住,只是少林僧众慈悲为本,不愿伤她性命,所出招数都非杀手,只求将她打倒,训诫一番,扣下兵刃,将她逐下山去。可是郭襄剑光错落,却也不易攻近身去。众僧初时只道一个妙龄女郎,还不轻易打发?待见她剑法精奇,始知她若非名门之女,便是名师之徒,多半得罪不得,出招时更有分寸,一面急报罗汉堂首座无色禅师。正斗之间,一个身材高瘦老年僧人缓步走近,双手笼在袖中,微笑观斗。两名僧人走到他身前,低声禀告了几句。郭襄已斗得气喘吁吁,剑法凌乱,大声喝道:“说甚么天下武学之源,原来是十多个和尚一拥而上,倚多为胜。”那老僧便是罗汉堂首座无色禅师,听她这么说,便道:“各人住手!”众僧人立时罢手跃开。无色禅师道:“姑娘贵姓,令尊和令师是谁?光临少林寺,不知有何贵干?”郭襄心道:“我爹娘的姓名不能告诉你。我到少林寺来是为了打听大哥哥的讯息,那也不能当众述说。眼下已闹成这等模样,日后爹娘和大哥哥知道了定要怪我,不如悄悄的溜了罢。”说道:“我的姓名不能跟你说,我不过见山上风景优美,这便上来游览玩耍。原来少林寺比皇宫内院还要厉害,动不动便要扣人家兵刃。请问大师,我进了贵寺的山门没有?当日达摩祖师传下武艺,想来也不过教众僧侣强身健体,便于精进修为,想不到少林寺名头越大,武功越高,恃众逞强的名头也越来越响。好,你们要扣我兵刃,这便留下,除非将我杀了,否则今日之事江湖上不会无人知晓。”她本来伶牙俐齿,这件事也并非全是她的过错,一席话只将无色禅师说得哑口无言。郭襄鉴貌辨色,心想:“这番胡闹我固怕人知晓,看来少林寺更加不愿张扬。十多个和尚围斗一个年轻姑娘,说出去有甚么好听?”当下哼的一声,将短剑往地下一掷,举步便行。"
    )
    val book = EBook("倚天屠龙记","张天爱","武侠小说")
    chapters.foreach(chapter=>book.addChapter(chapter))
    book.play()
  }
}

效果展示

“春游浩荡,是年年寒食,梨花时节。白锦无纹香烂漫,玉树琼苞堆
雪。静夜沉沉,浮光霭霭,冷浸溶溶月。人间天上,烂银霞照通彻。
浑似姑射真人,天姿灵秀,意气殊高洁。万蕊参差谁信道,不与群芳
同列。浩气清英,仙才卓荦,下土难分别。瑶台归去,洞天方看清绝
。”
  作这一首《无俗念》词的,乃南宋末年一位武学名家,有道之士
。此人姓丘,名处机,道号长春子,名列全真七子之一,是全真教中
出类拔萃的人物。《词品》评论此词道:“长春,世之所谓仙人也,
而词之清拔如此”。这首词诵的似是梨花,其实词中真意却是赞誉一
位身穿白衣的美貌少女,说她“浑似姑射真人,天姿灵秀,意气殊高
洁”,又说她“浩气清英,仙才卓荦”,“不与群芳同列”。词中所
颂这美女,乃古墓派传人小龙女。她一生爱穿白衣,当真如风拂玉树
,雪裹琼苞,兼之生性清冷,实当得起“冷浸溶溶月”的形容,以“
无俗念”三字赠之,可说十分贴切。长春子丘处机和她在终南山上比
邻而居,当年一见,便写下这首词来。
  这时丘处机逝世已久,小龙女也已嫁与神雕大侠杨过为妻。在河
南少室山山道之上,却另有一个少女,正在低低念诵此词。这少女十
**岁年纪,身穿淡黄衣衫,骑着一头青驴,正沿山道缓缓而上,心
中默想:“也只有龙姊姊这样的人物,才配得上他。”这一个“他”
字,指的自然是神雕大侠杨过了。她也不拉缰绳,任由那青驴信步而
行,一路上山。过了良久,她又低声吟道:“欢乐趣,离别苦,就中
更有痴儿女。君应有语,渺万里层云,千山暮雪,只影向谁去?”
是否继续读取:y
她腰悬短剑,脸上颇有风尘之色,显是远游已久;韶华如花,正当喜
乐无忧之年,可是容色间却隐隐有懊闷意,似是愁思袭人,眉间心上
,无计回避。
  这少女姓郭,单名一个襄字,乃大侠郭靖和女侠黄蓉的次女,有
个外号叫做“小东邪”。她一驴一剑,只身漫游,原想排遣心中愁闷
,岂知酒入愁肠固然愁上加愁,而名山独游,一般的也是愁闷徒增。
河南少室山山势颇陡,山道却是一长列宽大的石级,规模宏伟,工程
着实不小,那是唐朝高宗为临幸少林寺而开凿,共长八里。郭襄骑着
青驴委折而上,只见对面山上五道瀑布飞珠溅玉,奔泻而下,再俯视
群山,已如蚁蛭。顺着山道转过一个弯,遥见黄墙碧瓦,好大一座寺
院。
  她望着连绵屋宇出了一会神,心想:“少林寺向为天下武学之源
,但华山两次论剑,怎地五绝之中并无少林寺高僧?难道寺中和尚自
忖没有把握,生怕堕了威名,索性便不去与会?又难道众僧侣修为精
湛,名心尽去,武功虽高,却不去和旁人争强赌胜?”她下了青驴,
缓步走向寺前,只见树木森森,荫着一片碑林。石碑大半已经毁破,
字迹模糊,不知写着些甚么。心想:“便是刻凿在石碑上的字,年深
月久之后也须磨灭,如何刻在我心上的,却是时日越久反而越加清晰
?”瞥眼只见一块大碑上刻着唐太宗赐少林寺寺僧的御札,嘉许少林
寺僧立功平乱。碑文中说唐太宗为秦王时,带兵讨伐王世充,少林寺
和尚投军立功,最著者共一十三人。其中只昙宗一僧受封为大将军,
其余十二僧不愿为官,唐太宗各赐紫罗袈裟一袭。她神驰想象:“当
隋唐之际,少林寺武功便已名驰天下,数百年来精益求精,这寺中卧
虎藏龙,不知有多少好手。”郭襄自和杨过、小龙女夫妇在华山绝顶
分手后,三年来没得到他二人半点音讯。她心中长自记挂,于是禀明
父母,说要出来游山玩水,实则是打听杨过的消息,她倒也不一定要
和他夫妇会面,只须听到一些杨过如何在江湖上行侠的讯息,也便心
满意足了。偏生一别之后,他夫妇从此便不在江湖上露面,不知到了
何处隐居,郭襄自北而南,又从东至西,几乎踏遍了大半个中原,始
终没听到有人说起神雕大侠杨过的近讯。这一日她到了河南,想起少
林寺中有一位僧人无色禅师是杨过的好友,自己十六岁生日之时,无
色瞧在杨过的面上,曾托人送来一件礼物,虽然从未和他见过面,但
不妨去问他一问,说不定他会知道杨过的踪迹,这才上少林寺来。正
出神间,忽听得碑林旁树丛后传出一阵铁链当啷之声,一人诵念佛经
:“是时药叉共王立要,即于无量百千万亿大众之中,说胜妙伽他曰
:由爱故生忧,由爱故生怖;若离于爱者,无忧亦无怖……”郭襄听
了这四句偈言,不由得痴了,心中默默念道:“由爱故生忧,由爱故
生怖;若离于爱者,无忧亦无怖。”只听得铁链拖地和念佛之声渐渐
远去。郭襄低声道:“我要问他,如何才能离于爱,如何能无忧无怖
?”随手将驴缰在树上一绕,拨开树丛,追了过去。只见树后是一条
上山的小径,一个僧人挑了一对大桶,正缓缓往山上走去。郭襄快步
跟上,奔到距那僧人七八丈处,不由得吃了一惊,只见那僧人挑的是
一对大铁桶,比之寻常水桶大了两倍有余,那僧人颈中、手上、脚上
,更绕满了粗大的铁链,行走时铁链拖地,不停发出声响。这对大铁
桶本身只怕便有二百来斤,桶中装满了水,重量更是惊人。郭襄叫道
:“大和尚,请留步,小女子有句话请教。”
是否继续读取:n

Process finished with exit code 0

8、内部类

8.1:使用内部类的目的

1.封装代码:将代码进行封装,达到重用的效果。

2.隐藏实现:在内部类中通过【访问控制修饰符】来决定是否对外开放。

        若定义了一个【私有的内部类(private)】,那么【这个内部类就只能在外部类的内部使用,外部类的客户端无法直接访问或实例化这个内部类,从而实现了隐藏实现】。这种方式可以有效地封装类的实现细节,提高代码的安全性和可维护性。

3.易维护易扩展

8.2:具体讲解

知识点一:Scala中内部类是外部类对象的成员
/*
	Java中内部类是【外部类的成员】:
		InClass ic = new OutClass.InClass()
	Scala中内部类是【外部类对象的成员】:
		val out = new OutClass();
		val in = new out.InClass();
*/

//外部类
class OutClass {
  //内部类:隐藏实现
  class InClass{
    override def toString: String = "InClass"
  }
  private val in:InClass = new InClass()

  override def toString: String = s"OutClass{${in}}"
}


object ScalaOOP {
  def main(args: Array[String]): Unit = {
    //创建外部类对象
    val out = new OutClass
    println(out)
    //外部访问内部类 => 内部类是外部类对象的资源
    val in = new out.InClass
    println(in)
  }
}
--------------------------
OutClass{InClass}
InClass
--------------------------
知识点二:内部类推荐写入伴生对象中
import scala.cha06.OutClass.InClass
//外部类
class OutClass {
  private val in:InClass = new InClass()

  override def toString: String = s"OutClass{${in}}"
}
//内部类:【推荐】写到【伴生对象】中
object OutClass{
  class InClass{
    override def toString: String = "InClass"
  }
}

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    //创建外部类对象
    val out = new OutClass
    println(out)
    //外部访问内部类 => 内部类是外部类对象的资源
    val in = new OutClass.InClass
    println(in)
  }
}
--------------------------
OutClass{InClass}
InClass
--------------------------

9、样例类

9.1:样例类基本知识

样例类:
	1.默认状态下:【不可变值】的对象
		样例类构造参数【默认声明】为 val,自动生成 getter【不可变值】
		样例类的构造参数若声明为 var,自动生成 getter & setter【可变值】
	2.自动生成:
		样例类自动生成伴生对象
		样例类自动实现的其他方法:toString,copy,equals,hashCode
	3.【样例类伴生对象实现】的方法:apply, unapply(用于模式匹配)
	4.【注意点】:
		父类可以是样例类或普通类,但其叶子节点[最底层节点]不能是样例类。

9.2:样例类详细案例讲解

一:模式匹配案例
// 普通类的模式匹配案例
case class Student(name:String, age:Int)	// 构造参数默认 val
case class Point(var x:Int,var y:Int)		// var 需要显式声明

// 追加伴生对象并实现 apply & unapply 之后,不再报错
object Point{
  //形成对象
  def apply(x: Int, y: Int): Point = new Point(x, y)
  //拆解,提取对象中内容
  def unapply(arg: Point): Option[(Int, Int)] = Some((arg.x,arg.y)) 
}

val obj:Any = Student("张三",18)
val info = obj match {
    case Student(_,22) => "22"
    case Student(name,_) if name.startsWith("张") => "姓张"
    case Point(a,_) => s"$a" // 报错 cannot resolve symbol
}
-------------------
姓张
-------------------
二:验证可变值与不可变值案例
// 默认情况下val:【不可变值】的对象
object ScalaOOP {
  def main(args: Array[String]): Unit = {
    case class Point(x:Int,y:Int)
    val point: Point = Point(10, 20)
    println(point.x)//获取信息
  }
}
------------
10
------------

// var:可变值
object ScalaOOP {
  def main(args: Array[String]): Unit = {
    case class Point(var x:Int,var y:Int)
    val point: Point = Point(10, 20)
    point.x = 21
    println(point.x)
  }
}
------------
21
------------

10、枚举【了解】

/*
	单例对象通过继承 Enumeration 实现枚举创建,简单易用,但不可扩展
	通常用于定义一个有限取值范围的常量
*/
class Choice extends Enumeration {
  val FIRST_CHOICE = Value(0)
  val FIRST_SECOND = Value(1)
  val FIRST_THIRD = Value(2)
}

object ScalaOOP {
  def main(args: Array[String]): Unit = {
    val choice: Choice.Value = Choice.FIRST_CHOICE
    val num: Int = choice match {
      case Choice.FIRST_CHOICE => 11
      case Choice.SECOND_CHOICE => 12
      case Choice.THIRD_CHOICE => 13
    }
    println(num)
  }
}
------------
11
------------

11、泛型

11.1:泛型定义

泛型的定义类型参数化,主要用于【集合】。不同于 Java ,scala中泛型被定义在 [] 中

11.2:泛型简单体验

//泛型体验
class Many[T](t:T){
  val _t:T = t
  override def toString: String = _t.toString
}


object ScalaOOP {
  def main(args: Array[String]): Unit = {
    val many:Many[String] = new Many[String]("123")
    println(many)
  }
}
-------------
123
-------------

11.3:泛型的边界定义

class GrandFather(name:String) {
  val _name:String = name
  override def toString: String = _name
}
object GrandFather{
  def apply(name: String): GrandFather = new GrandFather(name)
}

class Father(name:String) extends GrandFather(name:String) {}
object Father{
  def apply(name: String): Father = new Father(name)
}

class Son(name:String) extends Father(name:String) {}
object Son{
  def apply(name: String): Son = new Son(name)
}

class GrandSon(name:String) extends Son(name:String){}
object GrandSon{
  def apply(name: String): GrandSon = new GrandSon(name)
}

/*
	泛型边界定义
		上边界:T<:A	泛型为某个类型的子类
		下边界:T>:A	泛型为某个类型的父类
*/
// 1.只能是比Father小的类型
class MyArray[T <: Father](items:T*) {
  def join(sep:String) = items.mkString(sep)
}
// Type GrandFather does not conform to 【 upper bound 】 Father of type parameter T
val arr:MyArray[GrandFather] = ...

=================================================================================

// 2.只能是比Son大的类型
class MyArray[T >: Son](items:T*) {
  def join(sep:String) = items.mkString(sep)
}
// Type GrandSon does not conform to 【 lower bound 】 Son of type parameter T
val arr:MyArray[GrandSon] = ...

11.4:泛型的多态

class GrandFather(name:String) {
  val _name:String = name
  override def toString: String = _name
}
object GrandFather{
  def apply(name: String): GrandFather = new GrandFather(name)
}

class Father(name:String) extends GrandFather(name:String) {}
object Father{
  def apply(name: String): Father = new Father(name)
}

class Son(name:String) extends Father(name:String) {}
object Son{
  def apply(name: String): Son = new Son(name)
}

class GrandSon(name:String) extends Son(name:String){}
object GrandSon{
  def apply(name: String): GrandSon = new GrandSon(name)
}

/*
	泛型的变化 => 影响泛型类的变化
	型变:多态
		协变:[+T]		若A是B的子类,则 C[A]为C[B]的子类 ✔【推荐】
		逆变:[-T]		若A是B的子类,则 C[B]为C[A]的子类
		不变:[T]		默认
*/
class MyArray[+T](items:T*) {
  def join(sep:String) = items.mkString(sep)
}
// Father 是 Son 的父类,则 MyArray[Father] 就是 MyArray[Son] 的父类
val arr:MyArray[Father] = new MyArray[Son](Son("henry"),Son("ariel"))

=================================================================================

class MyArray[-T](items:T*) {
  def join(sep:String) = items.mkString(sep)
}
// Father 是 Son 的子类,则 MyArray[Son] 就是 MyArray[Father] 的子类
val arr:MyArray[Son] = new MyArray[Father](Son("henry"),Son("ariel"))

=================================================================================

class MyArray[T](items:T*) {
  def join(sep:String) = items.mkString(sep)
}
// 所有泛型都必须为 Son
val arr:MyArray[Son] = new MyArray[Son](Son("henry"),Son("ariel"))

12、包与包对象

/*
 【包】
	包命名规则:字母、数字、下划线、点,不能以数字开头,在【一个类文件中可以定义多个并列的包】
	导包的不同方式
		import com.kgc.Person				方便使用类 Person
		import com.kgc._					方便使用 com.kgc 包中的所有类
		import com.kgc.Person._				方便使用类 Person 中的所有属性和方法
		import com.kgc.{Person=>PS,Book}	只导入包中 Person和Book,并将Person重命名为PS
	不同于Java
		import 导包语句可以出现在任意地方
		可以导入包、类、类成员
*/

// 单个文件多包结构:【资源】按【包名】语义【分类存放】,方便管理和使用
----------------------- Test.scala START ----------------------------
package cha03{
  import cha03.util.Sorts
  object PackageTest {
    def main(args: Array[String]): Unit = {
      val array: Array[Int] = Array(3, 1, 5, 4, 2)
      Sorts.insertSort(array)
      array.foreach(println)
    }
  }
}

package cha03.util{
  object Sorts{
    def insertSort(array: Array[Int]): Unit ={
      import scala.util.control.Breaks._
      for(i<- 1 until array.length){
        val t = array(i)
        var j = i-1
        breakable({
          while (j>=0){
            if(array(j)>t){
              array(j+1) = array(j)
            }else{
              break()
            }
            j-=1
          }
        })
        array(j+1) = t
      }
    }
  }
}
----------------------- Test.scala END ----------------------------

/*
 【包对象】
	包中可以包含:类、对象、特质...
	包对象可以包含:除了类、对象、特质外,还可以包含变量和方法
*/
package cha03{
  import cha03.util.Constants._
  import cha03.util.Constants.{DataFormat=>DF}
  
  object PackageTest {
    def main(args: Array[String]): Unit = {
      println(PI*2*2)
      println(getQuarter(5))
      val format: DF = DF(2024, 3, 29)
      println(format.stdYMD())
      println(format.stdFull())
      println(format.timestamp())
    }
  }
}
package cha03.util{
  import java.util.Calendar
  // 包对象
  package object Constants{
    // 变量
    val PI:Float = 3.14f
	// 方法
    def getQuarter(month:Int)=(month-1)/3+1
	// 类
    class DataFormat(
        year:Int,month:Int,day:Int,
        hour:Int,minute:Int,second:Int,
        millis:Int){
      private var _year:Int = year
      private var _month:Int = month
      private var _day:Int = day
      private var _hour:Int = hour
      private var _minute:Int = minute
      private var _second:Int = second
      private var _millis:Int = millis

      def this(year:Int,month:Int,day:Int){
        this(year,month,day,0,0,0,0)
      }

      def stdYMD():String = s"${_year}-${_month}-${_day}"
      def stdFull():String = s"${_year}-${_month}-${_day} ${_hour}:${_minute}:${_second}.${_millis}"
      def timestamp():Long = {
        val cld = Calendar.getInstance()
        cld.set(_year,_month,_day,_hour,_minute,_second)
        cld.set(Calendar.MILLISECOND,555)
        cld.getTimeInMillis
      }
    }
  }
  object DataFormat{
      def apply(year: Int, month: Int, day: Int, 
                hour: Int, minute: Int, second: Int, millis: Int): DataFormat 
      	= new DataFormat(year, month, day, hour, minute, second, millis)
      def apply(year: Int, month: Int, day: Int): DataFormat 
      	= new DataFormat(year, month, day)
  }
}

// 导包的形式
import pack_name						包本身
import pack_name._						包下面的所有资源
import pack_name.{A=>AS,B}				包下面部分资源
import pack_name.Class					类本身
import pack_name.Class.{PI,show}		类中的方法属性等
import pack_name.Class._				类下面的所有资源
  • 13
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值