优化阅读格式

This commit is contained in:
heibaiying
2019-07-31 17:18:07 +08:00
parent ceb868fe13
commit ca7c99802b
91 changed files with 4059 additions and 4058 deletions

View File

@ -20,50 +20,50 @@
### 1.1 Scala中的继承结构
Scala中继承关系如下图
Scala 中继承关系如下图:
+ Any是整个继承关系的根节点
+ AnyRef包含Scala ClassesJava Classes等价于Java中的java.lang.Object
+ AnyVal是所有值类型的一个标记
+ Null是所有引用类型的子类型唯一实例是null可以将null赋值给除了值类型外的所有类型的变量;
+ Nothing是所有类型的子类型。
+ Any 是整个继承关系的根节点;
+ AnyRef 包含 Scala ClassesJava Classes等价于 Java 中的 java.lang.Object
+ AnyVal 是所有值类型的一个标记;
+ Null 是所有引用类型的子类型,唯一实例是 null可以将 null 赋值给除了值类型外的所有类型的变量;
+ Nothing 是所有类型的子类型。
<div align="center"> <img src="https://github.com/heibaiying/BigData-Notes/blob/master/pictures/scala继承层次.png"/> </div>
### 1.2 extends & override
Scala的集成机制和Java有很多相似之处比如都使用`extends`关键字表示继承,都使用`override`关键字表示重写父类的方法或成员变量。示例如下:
Scala 的集成机制和 Java 有很多相似之处,比如都使用 `extends` 关键字表示继承,都使用 `override` 关键字表示重写父类的方法或成员变量。示例如下:
```scala
//父类
class Person {
var name = ""
// 1.不加任何修饰词,默认为public,能被子类和外部访问
// 1.不加任何修饰词,默认为 public,能被子类和外部访问
var age = 0
// 2.使用protected修饰的变量能子类访问但是不能被外部访问
// 2.使用 protected 修饰的变量能子类访问,但是不能被外部访问
protected var birthday = ""
// 3.使用private修饰的变量不能被子类和外部访问
// 3.使用 private 修饰的变量不能被子类和外部访问
private var sex = ""
def setSex(sex: String): Unit = {
this.sex = sex
}
// 4.重写父类的方法建议使用override关键字修饰
// 4.重写父类的方法建议使用 override 关键字修饰
override def toString: String = name + ":" + age + ":" + birthday + ":" + sex
}
```
使用`extends`关键字实现继承:
使用 `extends` 关键字实现继承:
```scala
// 1.使用extends关键字实现继承
// 1.使用 extends 关键字实现继承
class Employee extends Person {
override def toString: String = "Employee~" + super.toString
// 2.使用publicprotected关键字修饰的变量能被子类访问
// 2.使用 publicprotected 关键字修饰的变量能被子类访问
def setBirthday(date: String): Unit = {
birthday = date
}
@ -92,7 +92,7 @@ object ScalaApp extends App {
### 1.3 调用超类构造器
在Scala的类中每个辅助构造器都必须首先调用其他构造器或主构造器这样就导致了子类的辅助构造器永远无法直接调用超类的构造器只有主构造器才能调用超类的构造器。所以想要调用超类的构造器代码示例如下
Scala 的类中,每个辅助构造器都必须首先调用其他构造器或主构造器,这样就导致了子类的辅助构造器永远无法直接调用超类的构造器,只有主构造器才能调用超类的构造器。所以想要调用超类的构造器,代码示例如下:
```scala
class Employee(name:String,age:Int,salary:Double) extends Person(name:String,age:Int) {
@ -102,7 +102,7 @@ class Employee(name:String,age:Int,salary:Double) extends Person(name:String,age
### 1.4 类型检查和转换
想要实现类检查可以使用`isInstanceOf`,判断一个实例是否来源于某个类或者其子类,如果是,则可以使用`asInstanceOf`进行强制类型转换。
想要实现类检查可以使用 `isInstanceOf`,判断一个实例是否来源于某个类或者其子类,如果是,则可以使用 `asInstanceOf` 进行强制类型转换。
```scala
object ScalaApp extends App {
@ -117,7 +117,7 @@ object ScalaApp extends App {
// 2. 强制类型转换
var p: Person = employee.asInstanceOf[Person]
// 3. 判断一个实例是否来源于某个类(而不是其子类)
// 3. 判断一个实例是否来源于某个类 (而不是其子类)
println(employee.getClass == classOf[Employee])
}
@ -127,7 +127,7 @@ object ScalaApp extends App {
#### **1. 构造顺序**
在Scala中还有一个需要注意的问题如果你在子类中重写父类的val变量并且超类的构造器中使用了该变量那么可能会产生不可预期的错误。下面给出一个示例
Scala 中还有一个需要注意的问题,如果你在子类中重写父类的 val 变量,并且超类的构造器中使用了该变量,那么可能会产生不可预期的错误。下面给出一个示例:
```scala
// 父类
@ -151,7 +151,7 @@ object ScalaApp extends App {
}
```
这里初始化array用到了变量range这里你会发现实际上array既不会被初始化Array(10)也不会被初始化为Array(2),实际的输出应该如下:
这里初始化 array 用到了变量 range这里你会发现实际上 array 既不会被初始化 Array(10),也不会被初始化为 Array(2),实际的输出应该如下:
```properties
父类的默认构造器
@ -159,19 +159,19 @@ object ScalaApp extends App {
()
```
可以看到array被初始化为Array(0),主要原因在于父类构造器的执行顺序先于子类构造器,这里给出实际的执行步骤:
可以看到 array 被初始化为 Array(0),主要原因在于父类构造器的执行顺序先于子类构造器,这里给出实际的执行步骤:
1. 父类的构造器被调用,执行`new Array[Int](range)`语句;
2. 这里想要得到range的值会去调用子类range()方法,因为`override val`重写变量的同时也重写了其get方法
3. 调用子类的range()方法自然也是返回子类的range值但是由于子类的构造器还没有执行这也就意味着对range赋值的`range = 2`语句还没有被执行所以自然返回range的默认值也就是0。
1. 父类的构造器被调用,执行 `new Array[Int](range)` 语句;
2. 这里想要得到 range 的值,会去调用子类 range() 方法,因为 `override val` 重写变量的同时也重写了其 get 方法;
3. 调用子类的 range() 方法,自然也是返回子类的 range 值,但是由于子类的构造器还没有执行,这也就意味着对 range 赋值的 `range = 2` 语句还没有被执行,所以自然返回 range 的默认值,也就是 0。
这里可能比较疑惑的是为什么`val range = 2`没有被执行却能使用range变量这里因为在虚拟机层面是先对成员变量先分配存储空间并赋给默认值之后才赋予给定的值。想要证明这一点其实也比较简单代码如下:
这里可能比较疑惑的是为什么 `val range = 2` 没有被执行,却能使用 range 变量,这里因为在虚拟机层面,是先对成员变量先分配存储空间并赋给默认值,之后才赋予给定的值。想要证明这一点其实也比较简单,代码如下:
```scala
class Person {
// val range: Int = 10 正常代码 arrayArray(10)
// val range: Int = 10 正常代码 arrayArray(10)
val array: Array[Int] = new Array[Int](range)
val range: Int = 10 //如果把变量的声明放在使用之后此时数据arrayarray(0)
val range: Int = 10 //如果把变量的声明放在使用之后,此时数据 arrayarray(0)
}
object Person {
@ -186,15 +186,15 @@ object Person {
想要解决上面的问题,有以下几种方法:
(1) . 将变量用final修饰代表不允许被子类重写`final val range: Int = 10 `
(1) . 将变量用 final 修饰,代表不允许被子类重写,即 `final val range: Int = 10 `
(2) . 将变量使用lazy修饰代表懒加载即只有当你实际使用到array时候才去进行初始化
(2) . 将变量使用 lazy 修饰,代表懒加载,即只有当你实际使用到 array 时候,才去进行初始化;
```scala
lazy val array: Array[Int] = new Array[Int](range)
```
(3) . 采用提前定义代码如下代表range的定义优先于超类构造器。
(3) . 采用提前定义,代码如下,代表 range 的定义优先于超类构造器。
```scala
class Employee extends {
@ -208,13 +208,13 @@ class Employee extends {
但是这种语法也有其限制:你只能在上面代码块中重写已有的变量,而不能定义新的变量和方法,定义新的变量和方法只能写在下面代码块中。
>**注意事项**:类的继承和下文特质(trait)的继承都存在这个问题,也同样可以通过提前定义来解决。虽然如此,但还是建议合理设计以规避该类问题。
>**注意事项**:类的继承和下文特质 (trait) 的继承都存在这个问题,也同样可以通过提前定义来解决。虽然如此,但还是建议合理设计以规避该类问题。
<br/>
## 二、抽象类
Scala中允许使用`abstract`定义抽象类,并且通过`extends`关键字继承它。
Scala 中允许使用 `abstract` 定义抽象类,并且通过 `extends` 关键字继承它。
定义抽象类:
@ -227,7 +227,7 @@ abstract class Person {
// 2.定义抽象方法
def geDetail: String
// 3. scala的抽象类允许定义具体方法
// 3. scala 的抽象类允许定义具体方法
def print(): Unit = {
println("抽象类中的默认方法")
}
@ -254,10 +254,10 @@ class Employee extends Person {
### 3.1 trait & with
Scala中没有interface这个关键字想要实现类似的功能可以使用特质(trait)。trait等价于Java 8中的接口因为trait中既能定义抽象方法也能定义具体方法这和Java 8中的接口是类似的。
Scala 中没有 interface 这个关键字,想要实现类似的功能,可以使用特质 (trait)。trait 等价于 Java 8 中的接口,因为 trait 中既能定义抽象方法,也能定义具体方法,这和 Java 8 中的接口是类似的。
```scala
// 1.特质使用trait关键字修饰
// 1.特质使用 trait 关键字修饰
trait Logger {
// 2.定义抽象方法
@ -270,10 +270,10 @@ trait Logger {
}
```
想要使用特质,需要使用`extends`关键字,而不是`implements`关键字,如果想要添加多个特质,可以使用`with`关键字。
想要使用特质,需要使用 `extends` 关键字,而不是 `implements` 关键字,如果想要添加多个特质,可以使用 `with` 关键字。
```scala
// 1.使用extends关键字,而不是implements,如果想要添加多个特质可以使用with关键字
// 1.使用 extends 关键字,而不是 implements,如果想要添加多个特质,可以使用 with 关键字
class ConsoleLogger extends Logger with Serializable with Cloneable {
// 2. 实现特质中的抽象方法
@ -310,7 +310,7 @@ class InfoLogger extends Logger {
### 3.3 带有特质的对象
Scala支持在类定义的时混入`父类trait`,而在类实例化为具体对象的时候指明其实际使用的`子类trait`。示例如下:
Scala 支持在类定义的时混入 ` 父类 trait`,而在类实例化为具体对象的时候指明其实际使用的 ` 子类 trait`。示例如下:
<div align="center"> <img src="https://github.com/heibaiying/BigData-Notes/blob/master/pictures/scala带有特质的对象.png"/> </div>
@ -327,7 +327,7 @@ trait Logger {
trait ErrorLogger
```scala
// 错误日志打印继承自Logger
// 错误日志打印,继承自 Logger
trait ErrorLogger extends Logger {
// 覆盖空方法
override def log(msg: String): Unit = {
@ -339,7 +339,7 @@ trait ErrorLogger extends Logger {
trait InfoLogger
```scala
// 通知日志打印继承自Logger
// 通知日志打印,继承自 Logger
trait InfoLogger extends Logger {
// 覆盖空方法
@ -352,7 +352,7 @@ trait InfoLogger extends Logger {
具体的使用类:
```scala
// 混入trait Logger
// 混入 trait Logger
class Person extends Logger {
// 调用定义的抽象方法
def printDetail(detail: String): Unit = {
@ -361,12 +361,12 @@ class Person extends Logger {
}
```
这里通过main方法来测试
这里通过 main 方法来测试:
```scala
object ScalaApp extends App {
// 使用with指明需要具体使用的trait
// 使用 with 指明需要具体使用的 trait
val person01 = new Person with InfoLogger
val person02 = new Person with ErrorLogger
val person03 = new Person with InfoLogger with ErrorLogger
@ -377,30 +377,30 @@ object ScalaApp extends App {
}
```
这里前面两个输出比较明显,因为只指明了一个具体的`trait`,这里需要说明的是第三个输出,**因为trait的调用是由右到左开始生效的**,所以这里打印出`Error:scala`
这里前面两个输出比较明显,因为只指明了一个具体的 `trait`,这里需要说明的是第三个输出,**因为 trait 的调用是由右到左开始生效的**,所以这里打印出 `Error:scala`
### 3.4 特质构造顺序
`trait`有默认的无参构造器,但是不支持有参构造器。一个类混入多个特质后初始化顺序应该如下:
`trait` 有默认的无参构造器,但是不支持有参构造器。一个类混入多个特质后初始化顺序应该如下:
```scala
// 示例
class Employee extends Person with InfoLogger with ErrorLogger {...}
```
1. 超类首先被构造即Person的构造器首先被执行
1. 超类首先被构造,即 Person 的构造器首先被执行;
2. 特质的构造器在超类构造器之前,在类构造器之后;特质由左到右被构造;每个特质中,父特质首先被构造;
+ Logger构造器执行LoggerInfoLogger的父类
+ InfoLogger构造器执行
+ ErrorLogger构造器执行;
+ Logger 构造器执行LoggerInfoLogger 的父类);
+ InfoLogger 构造器执行;
+ ErrorLogger 构造器执行;
3. 所有超类和特质构造完毕,子类才会被构造。
<br/>
## 参考资料
1. Martin Odersky . Scala编程(第3版)[M] . 电子工业出版社 . 2018-1-1
2. 凯.S.霍斯特曼 . 快学Scala(第2版)[M] . 电子工业出版社 . 2017-7
1. Martin Odersky . Scala 编程 (第 3 版)[M] . 电子工业出版社 . 2018-1-1
2. 凯.S.霍斯特曼 . 快学 Scala(第 2 版)[M] . 电子工业出版社 . 2017-7