[TOC]

一、scala概述

1.1 简介

​ scala是一种多范式的编程语言,其设计的初衷是要集成面向对象编程和函数式编程的各种特性。Scala运行于Java平台(Java虚拟机),并兼容现有的Java程序。它也能运行于CLDC配置的Java ME中。目前还有另一.NET平台的实现,不过该版本更新有些滞后。Scala的编译模型(独立编译,动态类加载)与Java和C#一样,所以Scala代码可以调用Java类库(对于.NET实现则可调用.NET类库)。

1.2 scala安装和配置

​ scala运行于JVM之上,所以得先安装jdk,安装过程看之前的文章吧。先到下载scala安装包。这里我用的版本是 scala2.11.8。windows上安装基本就是点点点,这里就不演示了。安装好之后就默认就配置好了PATH环境变量。
​ 在Linux上安装的话,就下载对用的 .tgz包,然后解压到指定目录。配置PATH环境变量,这过程基本就是这样。最后在命令行下输入 scala,成功进入scala命令行即为成功(和Linux安装jdk一毛一样)

二、scala基础

要注意一点,就是在scala中,任何数据都是对象,可以通过数据本身调用一些方法,而在java中需要将数据赋值给一个引用变量之后,通过引用变量才能调用方法。如

"1".toInt()     这个在scala中是可以运行的,但是类似这样的方式不能在java中运行

2.1 常用数据类型

2.1.1数值类型:

Byte,Short,Int,Long,Float,Double

Byte:  8位有符号数字,从-128 到 127
Short: 16位有符号数据,从-32768 到 32767
Int:  32位有符号数据
Long: 64位有符号数据

例子:scala命令行中定义一个Int变量
val a:Byte = 10
a+10
得到:res9: Int = 20
这里的res9是新生成变量的名字

2.1.2 字符类型

Char和String,前者是单字符,后者是字符串

对于字符串,在scala中可以进行插值操作,如:
scala> val a="king"
a: String = king

scala> s"my name is ${a}"
res1: String = my name is king
注意:前面有个s;相当于执行:"My Name is " + s1

2.1.3 Unit类型

相当于java中的void类型,一般常用于函数方法的返回值类型

2.1.4 Nothing类型

一般是执行过程中,产生的异常exception的类型就是nothing

2.2 变量的声明和使用

var 变量名:类型=value
val 变量名:类型=value

其中类型可以省略,scala会根据value的类型自动推导变量的类型的。
例子:
var a:Int=8

var和val的区别:
val引用指向的内存地址不能改变,但是里面的内容可变。
var引用指向的内存地址可变。而且定义时,必须初始化
如:
scala> val a=2
a: Int = 2

scala> a=3
:12: error: reassignment to val
       a=3
        ^

scala> var b=2
b: Int = 2

scala> b=3
b: Int = 3
可以看到,没办法给a重新赋值

2.3 函数的初步使用

2.3.1 scala内置函数

scala有许多内置函数,可以直接使用,比如 scala.math 包下的各种数学函数

import scala.math._  导入math下的所有函数
scala> import scala.math._
import scala.math._

scala> max(2,3)
res0: Int = 3

2.3.2 自定义函数

def 函数名称 ([参数名:参数类型]*) : 返回值类型 = {}
例子:
1、求和

scala> def sum(x:Int,y:Int) : Int = x+y
sum: (x: Int, y: Int)Int

scala> sum(10,20)
res4: Int = 30

scala> var a = sum(10,20)
a: Int = 30

2、求阶乘,采用递归方式
scala> def myFactor(x:Int) : Int = {
| //实现
| if(x<=1)
| 1
| else
| x*myFactor(x-1)
| }
myFactor: (x: Int)Int

scala> myFactor(3)
res5: Int = 6

注意:没有return语句的话,函数的最后一句话就是函数的返回值。
上面函数有分支,1 和 x*myFactor(x-1) 都有可能是函数的最后一句话。

2.4 scala条件语句和循环语句

2.4.1 条件判断语句if/else:

if (判断条件) {}
else if (判断条件) {}
else {}

2.4.2循环语句:

for循环

//定义个列表,方便讲解操作for循环
var list = List("Mary","Tom","Mike")

println("-----for循环第一种写法-------")
for( s <- list) println(s)
// <-    代表scala中的提取符,把list中的每一个元素都提取出来,赋给s

println("-----for循环第二种写法-------")
//打印名字长度大于3 加判断。可以加上判断
for{
    s <- list
    if(s.length > 3)
} println(s)

println("-----for循环第三种写法-------")
//打印名字长度小于等于3 加判断
for(s <- list if s.length <= 3 ) println(s)

println("-----for循环第四种写法-------")
//使用yield关键字 作用:产生一个新的集合
//把list中的每个元素都变成大写,返回一个新的集合。
var newList = for{
    s <- list
    s1 = s.toUpperCase //把名字变成大写
} yield (s1)  //使用yield将处理后的元素变成新的集合

for( s <- newList) println(s)

while循环:

println("-----while循环写法-------")
    //定义一个循环变量
    var i = 0
    while(i < list.length){
      println(list(i))
      //自增
      //i++ 这种自增在scala是不可行的
      i += 1
    }

println("-----do while循环写法-------")
    //定义循环变量
    var j = 0
    do{
      println(list(j))
      j+=1
    } while (j < list.length)

foreach函数迭代:

println("-----foreach用法-------")
    list.foreach(println)//相当于 for( s <- list) println(s)

    /**
      * foreach说明
      *
      * foreach相当于循环
      * list.foreach(println) 使用了高阶函数(函数式编程)
      *
      * 还有一个循环 map
      * foreach和map的区别:
      * foreach没有返回值,map有返回值
      * spark中类似。
      */
第二种foreach写法:
list.foreach{
    case xxxxx
}
直接内部使用case进行模式匹配

2.4.3 嵌套循环和break语句

/**
      * 题目:判断101-200之间有多少个素数
      *
      * 程序分析:
      * 判断素数的方法:
      * 用一个数分别去 除 2 到sqor(这个数),如果能被整除,则表明此数不是素数,反之是素数
      * 举例:101  2~根号(101)
      *
      * 程序实现方法:
      * 定义两层循环
      *   第一层 101-200
      *     第二层 2 - 根号(第一层)
      *       判断能被整除,则不是素数
      *
      */

    println("-----循环嵌套-------")
    var count : Int = 0 //保存素数的个数
    var index_outer : Int = 0 //外层循环变量
    var index_inner : Int = 0 //内层循环变量
    //until 语句,x until y表示x到y,不包括y
    for (index_outer <- 101 until 200 ){
      var b = false  //标示是否可以被整除

      breakable{
        index_inner = 2
        while (index_inner <= sqrt(index_outer)){
          if(index_outer % index_inner == 0){
            //可以被整除
            b = true
            break
          }
          index_inner += 1
        }
      }

      if (!b){
        count +=1
      }
    }
    println("个数为: " + count)

break用法:

将需要break的语句块用 breakable{} 包含起来,然后里面使用break语句进行break

2.4.4 嵌套循环实现冒泡排序

/**
      * 冒泡排序
      * https://www.cnblogs.com/morethink/p/8419151.html#%E5%86%92%E6%B3%A1%E6%8E%92%E5%BA%8F
      *
      * 算法分析:
      * 1、比较相邻元素,如果第一个比第二个大,就交换位置
      * 2、对每一对相邻元素做相同的工作,这一步做完后,最后的元素会是最大的数
      * 3、针对所有元素,重复以上步骤。
      *
      * 程序分析:
      * 1、两层循环
      * 2、外层循环控制比较的次数
      * 3、内层循环控制到达位置,也就是结束比较的位置
      */

println("------------冒泡排序--------------")
    var listSort = LinkedList(3,9,1,6,5,7,10,2,4)
    var startIndex:Int = 0
    var secondIndex:Int = 0
    var tmp:Int = 0
    for (startIndex <- 0 until(listSort.length - 1)) {
      for (secondIndex <- startIndex + 1 until(listSort.length)) {
        if (listSort(startIndex) > listSort(secondIndex)) {
          tmp = listSort(startIndex)
          listSort(startIndex) = listSort(secondIndex)
          listSort(secondIndex) = tmp
        }
      }
    }

2.5 scala函数的参数

2.5.1 函数参数的求值策略

call by value:

对函数实参求值,并且只求一次
例子:
def test(x:Int)

call by name:

函数的实参在函数体内部用到的时候,才会被求值
例子:
def test(x:=>Int)
注意变量名和类型之间的符号,多了个 => ,不要漏了

可能这么讲,还不清楚,下面看看例子

scala> def test1(x:Int,y:Int) : Int = x+x
test1: (x: Int, y: Int)Int

scala> test1(3+4,8)
res0: Int = 14

scala> def test2(x: => Int,y: => Int) : Int = x+x
test2: (x: => Int, y: => Int)Int

scala> test2(3+4,8)
res1: Int = 14

执行过程对比:
test1  -->  test1(3+4,8)    --> test1(7,8)  --> 7+7 --> 14

test2  -->  test2(3+4,8)    --> (3+4)+(3+4) --> 7+7 --> 14
注意这里,在没用到实参之前,实参一直是3+4,而不是7

下面来一个更明显的例子:

//x 是 call by value , y 是 call by name
def bar(x:Int,y: => Int) : Int = 1

定义一个死循环函数
def loop() : Int = loop

调用 bar 函数:
1、bar(1,loop)
2、bar(loop,1)

哪种方式会产生死循环?
第二种方式
解析:
1、y 每次在函数中用到时,才会被求值,bar函数没有用到y,所以不会调用loop。
2、x call by value ,对函数参数求值(无论是否用到),并且只求一次,所以会产生死循环。

2.5.2 函数参数类型

1、默认参数

当你没有给参数赋值的时候,就使用默认值。
scala> def fun1(name:String="Tom") : String = "Hello " + name
fun1: (name: String)String

scala> fun1("Andy")
res0: String = Hello Andy

scala> fun1()
res1: String = Hello Tom

2、代名参数

过代名参数可以确定给哪个参数赋值。
scala> def fun2(str:String="Good Morning ", name:String="Tom ", age:Int=20)=str + name + " and the age is " + age
fun2: (str: String, name: String, age: Int)String

scala> fun2()
res2: String = Good Morning Tom  and the age is 20

//这里指定给哪个默认参数赋值
scala> fun2(name="Mary ")
res3: String = Good Morning Mary  and the age is 20

3、可变参数

类似于java中的可变参数,即 参数数量不固定,就是在普通参数后面加个 *

例:
求多个数字的和:
def sum(args:Int*) = {
   var result = 0
   for(s <- args) result += s
   result
}

//这里就是可变参数
scala> def sum(args:Int*) = {
| var result = 0
| for(s <- args) result += s
| result}
sum: (args: Int*)Int

scala> sum(1,2,4)
res4: Int = 7

scala> sum(1,2,4,3,2,5,3)
res5: Int = 20

2.6 懒值lazy

定义:如果常量(也就是val定义的)是lazy的,他的初始化会被延迟,推迟到第一次使用该常量的时候,例子:

scala> val x : Int = 10
x: Int = 10

scala> val y:Int = x+1
y: Int = 11

y 不是 lazy,定义后立即触发计算

scala> lazy val z : Int = x+1
z: Int = 

z 是lazy,初始化会被推迟,定义时不会出发计算。

scala> z
res6: Int = 11

当我们第一次使用z的时候,才会触发计算。

扩展:

Spark的核心是RDD(数据集合),Spark提供很多方法,操作RDD,算子。
算子分为两种:
1、Transformation : 延时加载,不会触发计算
2、Action : 会触发计算

再看一个例子:

(1)读一个存在的文件

    scala> lazy val words = scala.io.Source.fromFile("H:\\tmp_files\\student.txt").mkString
    words: String = 

    scala>  words
    res7: String =
    1       Tom     12
    2       Mary    13
    3       Lily    15

    (2)读一个不存在的文件
    scala> val words = scala.io.Source.fromFile("H:\\tmp_files\\studen1231312312t.txt").mkString
    java.io.FileNotFoundException: H:\tmp_files\studen1231312312t.txt (系统找不到指定的文件。)
      at java.io.FileInputStream.open0(Native Method)
      at java.io.FileInputStream.open(FileInputStream.java:195)
      at java.io.FileInputStream.(FileInputStream.java:138)
      at scala.io.Source$.fromFile(Source.scala:91)
      at scala.io.Source$.fromFile(Source.scala:76)
      at scala.io.Source$.fromFile(Source.scala:54)
      ... 32 elided
    会产生异常

    scala> lazy val words = scala.io.Source.fromFile("H:\\tmp_files\\studen1231312312312t.txt").mkString
    words: String = 
    如果是懒值不会产生异常,因为定义的时候并没有执行,所以不会产生异常

2.7 异常exception

和java类似,使用try catch finally捕捉和处理异常

try {}
    catch {
        case ex: exception_type1 => {
            异常处理代码
        }

        case ex: exception_type2 => {
            异常处理代码
        }

        case _:Exception => {
            这里是包含所有exception,如果上面的没有匹配到就到这里来了
        }
    } finally {
        xxxx
    }

2.8 数组

数组类型:

Array[T](N)定长数组,需指定数组长度
ArrayBuffer 可变长数组,需要另外导入包import scala.collection.mutable._

Array操作:

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

初始化赋值,默认值0

scala> val b = new Array[String](15)
b: Array[String] = Array(null, null, null, null, null, null, null, null, null, null, null, null, null, null, null)

scala> val c = Array("Tom","Mary","Andy")
c: Array[String] = Array(Tom, Mary, Andy)

scala> val c = Array("Tom","Mary",1)
c: Array[Any] = Array(Tom, Mary, 1)

未声明 Array类型,直接赋值,Array中是Any类型,即任何类型都可以

scala> val c:Array[String]=Array("Tom","Andy",1)
:11: error: type mismatch;
found   : Int(1)
required: String
val c:Array[String]=Array("Tom","Andy",1)

当声明 Array[String] 时,数组元素必须全为String
^

scala> val c:Array[String]=Array("Tom","Andy")
c: Array[String] = Array(Tom, Andy)

ArrayBuffer操作:

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

mutable代表可变

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

往数组中添加元素
scala> d += 1
res8: d.type = ArrayBuffer(1)

访问数组指定下标的元素:d(index)

删除指定元素(不是删除指定下标的元素):
scala> y-=3
res6: scala.collection.mutable.ArrayBuffer[Int] = ArrayBuffer(1)

数组常见操作:

1、遍历数组:
scala> var a = Array("Tom","Andy","Mary")
a: Array[String] = Array(Tom, Andy, Mary)

scala> for(s<-a) println(s)
Tom
Andy
Mary

scala> a.foreach(println)
Tom
Andy
Mary

2、求最大和最小值
scala> val myarray = Array(1,2,7,8,10,3,6)
myarray: Array[Int] = Array(1, 2, 7, 8, 10, 3, 6)

scala> myarray.max
res16: Int = 10

scala> myarray.min
res17: Int = 1

3、排序
表示使用某个比较函数进行排序
scala> myarray.sortWith(_>_)
res18: Array[Int] = Array(10, 8, 7, 6, 3, 2, 1)

scala> myarray.sortWith(_<_)
res19: Array[Int] = Array(1, 2, 3, 6, 7, 8, 10)

解释:myarray.sortWith(_>_)
完整:myarray.sortWith((a,b)=>{if(a>b) true else false})
(a,b)=>{if(a>b) true else false} 是匿名函数,没有名字,传入两个参数 a b,返回值是bool
sortWith(_>_) 是高阶函数,即参数是函数

多维数组:

和Java一样,通过数组的数组来实现

定义一个固定长度的二维数组

scala> val matrix = Array.ofDim[Int](3,4)
matrix: Array[Array[Int]] = Array(
Array(0, 0, 0, 0), 
Array(0, 0, 0, 0), 
Array(0, 0, 0, 0)
)

scala> matrix(1)(2)=10

scala> matrix
res21: Array[Array[Int]] = Array(
Array(0, 0, 0, 0), 
Array(0, 0, 10, 0), 
Array(0, 0, 0, 0))

定义一个二维数组,其中每个元素都是一个一维数组,其长度不固定
scala> var triangle = new Array[Array[Int]](10)
triangle: Array[Array[Int]] = Array(null, null, null, null, null, null, null, null, null, null)

scala> for(i <- 0 until triangle.length)
| triangle(i)=new Array[Int](i+1)

scala> triangle
res23: Array[Array[Int]] = Array(
Array(0), 
Array(0, 0), 
Array(0, 0, 0), 
Array(0, 0, 0, 0), 
Array(0, 0, 0, 0, 0), 
Array(0, 0, 0, 0, 0, 0), 
Array(0, 0, 0, 0, 0, 0, 0), 
Array(0, 0, 0, 0, 0, 0, 0, 0), 
Array(0, 0, 0, 0, 0, 0, 0, 0, 0), 
Array(0, 0, 0, 0, 0, 0, 0, 0, 0, 0))

2.9 映射Map

Map[K,V]() 泛型类型可以根据传入的值自动确定,如果创建为空的话,KV都是nothing类型
scala.collection.mutable.Map  是可变的Map,添加进去的KV可更改
scala.collection.immutable.Map  是不可变Map

例子:
// 初始化赋值方式1
scala> val scores = Map("Tom" -> 80, "Mary"->77,"Mike"->82)
scores: scala.collection.mutable.Map[String,Int] = Map(Mike -> 82, Tom -> 80, Mary -> 77)

// 初始化赋值方式2
scala> val chineses = Map(("Tom",80),("Mary",60),("Lily",50))
chineses: scala.collection.mutable.Map[String,Int] = Map(Tom -> 80, Lily -> 50, Mary -> 60)

映射的操作:

1、获取映射中的值
scala> chineses("Tom")
res25: Int = 80

scala> chineses("To123123m")
java.util.NoSuchElementException: key not found: To123123m
at scala.collection.MapLike$class.default(MapLike.scala:228)
at scala.collection.AbstractMap.default(Map.scala:59)
at scala.collection.mutable.HashMap.apply(HashMap.scala:65)
... 32 elided

解决:先判断key是否存在
if(chineses.contains("To123123m")){
chineses("To123123m")
} else {
1
}

scala> if(chineses.contains("To123123m")){
| chineses("To123123m")
| } else {
| 1}
res27: Int = 1

//获取指定key的value,如果key不存在就返回-1,类似于返回默认值
scala> chineses.getOrElse("To123123m",-1)
res28: Int = -1

//get方法如果对应的key不存在,value返回为none
scala> chineses.get("dlfsjldkfjlsk")
res29: Option[Int] = None

使用get方法不会报错

scala> chineses.get("Tom")
res30: Option[Int] = Some(80)

Option   None    Some   三个类型,在后面做同时的讲解。

==========================================================
2、更新映射中的值
scala> chineses
res31: scala.collection.mutable.Map[String,Int] = Map(Tom -> 80, Lily -> 50, Mary -> 60)

scala> chineses("Tom")
res32: Int = 80

scala> chineses("Tom")=100

scala> chineses
res34: scala.collection.mutable.Map[String,Int] = Map(Tom -> 100, Lily -> 50, Mary -> 60)

==========================================================
3、映射的迭代
可以使用 for foreach
scala> chineses
res35: scala.collection.mutable.Map[String,Int] = Map(Tom -> 100, Lily -> 50, Mary -> 60)

scala> for(s <- chineses) println(s)
(Tom,100)
(Lily,50)
(Mary,60)

scala> chineses.foreach(println)
(Tom,100)
(Lily,50)
(Mary,60)

foreach本质是一个 高阶函数。

==========================================================
4、添加值到映射中
scala> a+="tom"->"king"  前面是K,后面是V

2.10元祖Tuple

元组是不可变类型,不能添加元素和修改元素。Scala 的 Tuple : 是不同类型值的集合

Tuple的声明:   
scala> val t1 = Tuple(1,0.3,"Hello")
:14: error: not found: value Tuple
val t1 = Tuple(1,0.3,"Hello")
    ^

scala> val t1 = Tuple3(1,0.3,"Hello")
t1: (Int, Double, String) = (1,0.3,Hello)

Tuple3 代表 Tuple 中有三个元素

可以使用下面的方式来定义任意元素个数的元祖
scala> val t1 = (1,0.3,"Hello")
t1: (Int, Double, String) = (1,0.3,Hello)

scala> val t1 = (1,0.3,"Hello",1,12,5,"all")
t1: (Int, Double, String, Int, Int, Int, String) = (1,0.3,Hello,1,12,5,all)

访问元素:注意这个特殊的符号
scala> t1._1   访问元祖中的第一个元素
res38: Int = 1

scala> t1._3   访问元祖中的第3个元素
res39: String = Hello

遍历tuple:

注意:Tuple中没有提供foreach函数,我们需要使用 productIterator   
遍历分成2个步骤:
1、使用productIterator 生成迭代器
2、遍历

scala> t1.productIterator.foreach(println)
1cala>
0.3
Hello
1
12
5
all

2.11 文件操作

//读取文件,source本身是一个按字符存储的可迭代对象,默认是单个字符的
val source = scala.io.Source.fromFile(fileName, 字符集)
//将整个文件内容读取成一个字符串
source.mkString
//按行读取,一次一行
source.getLines()
//按单个字符读取
for(c<-source) {
    println(c)
}

//从URL中读取数据内容
val source2 = scala.io.Source.fromFile(URL, UTF-8)

//读取二进制文件,scala没有自己实现相应的类,所以是直接调用java的类
val file=new File(filename)
val in=new FileInputStream(file)
val buffer=new Array[Byte](file,.length().toInt)
in.read(buffer)

//写入文件
val  out=new PrintWriter(filename)
out.println(xxxxx) 写入内容

使用properties类解析resources目录下的properties配置文件:

们可以自定定义个 xx.properties 配置文件,然后使用load方法来读取。
配置文件的基本格式为  key=value 的格式,模块会自定解析成kv格式。

例子:
object Dataloader {
  def main(args: Array[String]): Unit = {
    val properties = new Properties()
    //注意这里因为要采用编译后的文件路径,所以这里是获取resource目录下文件在编译后的路径的输入流
    val propertiesStream = Dataloader.getClass.getClassLoader.getResourceAsStream("dataloader.properties")
    properties.load(propertiesStream)
    println(properties.getProperty("spark.local.cores"))

  }
}

三、scala面向对象特性

3.1 类的定义

scala中类的定义和java类似,同样使用class关键字定义类,但是class关键字前面不存在public等这些修饰符。例子:

class Student1 {
  //定义学生的属性
  private var stuId : Int = 0
  private var stuName : String = "Tom"
  private var age : Int = 20

  //定义成员方法(函数) get set
  def getStuName() : String = stuName
  def setStuName(newName :String) = this.stuName = newName

  def getStuAge() : Int = age
  def setStuAge(newAge : Int) = this.age = newAge
}

实际上scala会给属性自动生成对应的get和set方法,但是要注意以下原则:

1、默认情况下,如果属性没有用任何修饰符修饰,那么默认是private,但是自动生成的get和set方法是公有的.

2、而如果显式的声明属性为私有(显式使用private关键字),如:private var a = 10,那么生成的get和set方法都是私有的。只能在伴生对象中使用,或者在本类中使用

3、如果使用 private [this] var a = 10,这样定义属性,表示该属性完全不能被外部访问(包括伴生类),且不会生成set和get方法

4、如果值希望scala生成get方法,不生成set方法,可以将其定义为常量,因为常量的值不可改变
================== 本质 ===================
自动生成的set和get方法名是和属性名一致的,如:
var student = new Student1()
student.stuId   这里实际上是直接调用该属性的get方法的,只不过方法同名而已
student.age=10  这里实际上调用属性的set方法的

3.2 内部类

scala的内部类没有java这么复杂,就是简单在外部类中,直接定义一个类而已。例子:

class Student2 {

  //定义学生的属性
  private var stuName : String = "Tom"
  private var stuAge : Int = 20

  //定义一个数组保存学生的课程成绩
  private var courseList = new ArrayBuffer[Course]()

  //定义一个函数,用于添加学生的课程成绩
  def addNewCourse(cname:String,grade:Int) = {
    //创建课程的成绩信息
    var c = new Course(cname,grade)

    //添加到学生的对象中
    courseList += c

  }
  //定义课程类 主构造器就是写在类的后面
  class Course(var courseName:String,var grade:Int){
    //定义属性

    //定义函数
  }

}

可以通过外部类的方法来获取内部类的对象。
或者通过以下方式创建内部类对象
val test = new InnerClassTest() 只有当外部类对象是用val 定义时才可以
var testIn = new test.myClass("king")

注意,下面的方式定义是会报错的
var test = new InnerClassTest()   这里是var
var testIn = new test.myClass("king")

个人理解是:
scala中,内部类是属于外部类的对象的,不是属于外部类的(这是官方的话)
所以内部类对象也是跟随这外部类对象。要是外部类对象用var定义,表示其是可变的
要是它变了,内部类对象还怎么通过外部类对象的引用获取自己?

3.3 类的构造器

3.3.1 主构造器

主构造器就是在定义类的后面同时定义的,如:

class Student3(var stuName : String , var age:Int) 
括号里面的就是主构造器,只能是一些属性。

要注意一点,就是,括号里面定义的属性变量,前面的var或者val千万不能丢,如果丢掉了,则该参数只能被当做一个类内不可变参数使用,不能被当做类的字段,既外部不能访问该变量。就会造成以下后果,看例子:

//首先两个类,有主构造器,有var和没var 的区别
class a(var name:String)
class b(name:String)

object IODemo {
  def main(args: Array[String]): Unit = {
    //这里是可以正常创建对象的
    val king = new a("king")
    val wang = new b("wang")

    //到这里就有问题了
    king.name  这里是可以的
    wang.name  这会无法访问name这个属性,因为压根没有,只是作为class中的一个普通变量而已
  }
}

还有一个点,就是如果类里面有任何代码没有包含在类中的任何方法中时,实际上在创建类对象时,这些代码是会被执行的,所以其实也算是主构造器的一部分。如:
class a(var name:String){
    println("hahha")  //这个语句在创建对象时是会执行的  

    def test()={
        println("test")
    }
}

3.3.2 辅助构造器

一个类可以有多个辅助构造器,通过关键字 this 来实现,如:
class Student3(var stuName : String , var age:Int) {
  //属性
  private var gender:Int = 1

  //定义一个辅助构造器,辅助构造器可以有多个
  //辅助构造器就是一个函数,只不过这个函数的名字叫 this

  def this (age : Int){
    this("Mike",age) // new Student3("Mike",age)
    println("这是辅助构造器")
  }

  def this (){
    this(10)// new Student3("Mike",10)
    println("这是辅助构造器2")
  }

}

3.4 object对象(伴生对象)

object对象是scala中比较特殊的对象,有几个特性

1、Object中的内容都是静态的。所以里面定义的属性,方法都是静态的,可以直接通过类名调用,无需创建对象
2、scala中,没有static关键字,所有static都定义在object中。比如main函数
3、如果class的名字,跟object的名字一样,就把这个object叫做类的伴生对象。伴生对象。
4、main函数需要写在object中,但是不一定必须写在伴生对象中
5、本质上来说,object其实类似于一个单例对象,本身就是静态,不需要实例化对象来调用里面定义的方法和属性。
所以从这点上,可以算是弥补了scala没有静态属性的一个缺陷

使用object对象的单例对象特性的例子:

1、生成信用卡号
object CreditCard {

  //定义一个变量保存信用卡卡号
  private [this] var creditCardNumber : Long = 0

  //定义函数来产生卡号
  def generateCCNumber():Long = {
    creditCardNumber += 1
    creditCardNumber
  }

  //测试程序
  def main(args: Array[String]): Unit = {
    //产生新的卡号
    println(CreditCard.generateCCNumber())
    println(CreditCard.generateCCNumber())
    println(CreditCard.generateCCNumber())
    println(CreditCard.generateCCNumber())
    println(CreditCard.generateCCNumber())
    println(CreditCard.generateCCNumber())
    println(CreditCard.generateCCNumber())
}

}

2、扩展使用,App这个类
定义object类,继承App这个类,可以省略main函数,默认object中的所有代码都在main函数里面。如:
object AppTest extends App {
  println("test")   这个会直接执行
}

3.5 apply方法

说到这里,前面的估计会有有疑问,为什么有时候创建对象时,需要用new关键字,有时候又不需要了呢?这就要说到apply这个方法了。如:

var t1 = Tuple3(1,0.1,"Hello")

可以看到,创建一个tuple对象,但是并没有用new关键字,其实省略new 关键字时,是调用了该类的伴生对象中的apply方法。apply方法一般会返回创建好的类的对象,要注意以下几点:

1、使用apply方法,让程序更加简洁。
2、apply方法必须写在伴生对象中。因为直接调用apply方法需要是静态的,只能写在object伴生对象中
3、这种方式类似于通过类名.method 的方式来创建对象,
用java的方式来理解的话,即object是一个静态类,可以直接使用类名
调用里面的静态方法
4、而且apply方法内部,其实也是使用了 new 来创建对应类的对象的

例子:

//定义一个原始类
class Student4 (var stuName : String)

//定义上面类的伴生对象
object Student4{
  //定义apply方法
  def apply(name : String) = {
    println("调用apply方法")
    //返回原始类的对象
    new Student4(name)
  }

  //测试程序
  def main(args: Array[String]): Unit = {
    //通过主构造器创建学生对象
    var s1 = new Student4("Tom")
    println(s1.stuName)

    //通过apply方法创建学生对象,可以不写new关键字
    var s2 = Student4("Mary")
    println(s2.stuName)

  }
}

3.6 继承

3.6.1 普通继承

scala中,也是使用extends关键字继承父类的,而且也只允许单继承,如:

//定义父类
class Person(val name: String,val age:Int){
  //定义函数
  def sayHello() : String = "Hello " + name + " and the age is " + age
}

//定义子类
/**
  *class Emplyee(val name:String,val age:Int,val salary:Int) extends Person(name,age)
  * 上述写法报错
  * 如果想用子类的属性,覆盖父类的属性。直接这样写不行 需要override
  *
  *override就是希望使用子类中的值 覆盖父类中的值
  *
  */
class Emplyee(override val name:String,override val age:Int,val salary:Int) extends Person(name,age){
    //重写父类的方法
  override def sayHello(): String = "子类中的sayHello"
}

覆盖父类的方法或者属性时,都需要在最前面加上override关键字

3.6.2 匿名子类继承

object Demo1 {
  def main(args: Array[String]): Unit = {

    //创建Person对象
    var p1 = new Person("Tom",20)
    println(p1.name+"\t"+p1.age)
    println(p1.sayHello())

    //创建Employee对象
    var p2 : Person = new Emplyee("Mike",25,1000)
    println(p2.sayHello())

    //通过匿名子类来实现继承:没有名字的子类,就叫匿名子类
    var p3 : Person = new Person("Mary",25){
      override def sayHello(): String = "匿名子类中的sayHello"
    }
    println(p3.sayHello())
  }
}

3.6.3 抽象类和抽象字段

抽象类其实就是使用abstract关键字定义的类,如:

abstract class Teacher {
  var i:Int
  var name:String="king"
  def test = {

  }

  def sayHello

  def talk:Int
}

其中,有以下几点注意
1、抽象类中,没有初始化的字段为称为抽象字段,或者叫属性,如果初始化了就不是抽象字段了。同样的,没有实现的方法就称为抽象方法。
2、抽象类只能用于继承,不能实例化对象
3、非抽象子类中,继承了抽象父类,必须初始化抽象字段以及实现抽象方法。
初始化抽象字段时,可以在子类的主构造器或者在类实现中初始化也可以。但是在主构造器中可以不用立马初始化,可以等实例化对象时传入参数进行初始化。而在类实现中则必须立马初始化
4、抽象类中对于抽象属性,只会自动生成对应的get方法,不会生成set方法
5、继承抽象父类也是使用extends关键字。

3.7 trait特质

在scala中,一般只支持单继承,但是有时候需要多继承,所以就出现了trait,它可以让子类多继承。trait的定义基本和抽象类类似。如:

//定义父类
trait Human{
  //定义抽象属性
  val id : Int
  val name : String
}

//定义代表动作的trait
trait Action{
  //定义一个抽象函数
  def getActionName():String
}

//定义子类 从两个父类继承
class Student5(val id:Int,val name:String) extends Human with Action{
  override def getActionName(): String = "Action is running"
}

从上面也可以看到,继承的trait的时候,用到了extends + with 两个关键字,当父类多于一个时,则必须使用with关键字继承,且该类必须是trait。而extends后面的类可以是trait或者其他class类。多重继承如下:

class Father extends class1 with trait1 with trait2 with 。。。。。。{

}

3.8 包和包对象

scala包的使用和java类似,而且scala中,import语句可以在任何地方。一般情况下,java和scala的包下面,只能有类,对象,特质,不能直接定义函数以及变量。而scala的包对象可以解决这个问题。Scala中的包对象可以包含:常量,变量,方法,类,对象,trait(特质)。如:

package object MyPackage {
  def test = {

  }

  var x:Int = 0

  class a {}
}

然后可以直接通过包对象名直接调用里面的方法和变量了

四、scala中的集合

scala中集合一般分为可变集合和不可变集合,分别在 scala.collection.mutable(可变)和scala.collection.immutable(不可变)包下。不可变集合中不允许添加元素,不允许更改某个元素的值,允许删除元素。

4.1 列表List

List[T]() 不可变列表
LinkedList[T]() 可变列表

List[T]相关操作:

scala> val nameList = List("Tom","Andy")
nameList: List[String] = List(Tom, Andy)

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

空列表
scala> val nullList : List[Nothing] = List()
nullList: List[Nothing] = List()

二维列表
scala> val dim:List[List[Int]] = List(List(1,2,3),List(10,20))
dim: List[List[Int]] = List(List(1, 2, 3), List(10, 20))

返回第一个元素
nameList.head

返回除去最后一个元素的列表
nameList.tail

访问指定index 的元素
nameList(index)

LinkedList[T]相关操作:

定义可变列表
scala> val myList = scala.collection.mutable.LinkedList(1,2,3,4,5)
    warning: there was one deprecation warning; re-run with -deprecation for details
    myList: scala.collection.mutable.LinkedList[Int] = LinkedList(1, 2, 3, 4, 5)

遍历修改列表
var cur = myList  指向nameList的内存地址引用

    while(cur != Nil){
    cur.elem = cur.elem*2  迭代每个元素,并将每个元素 *2
    cur = cur.next
    }

col:+ ele
//将元素的添加到集合的尾部(seq)
ele +:col
//将元素添加到集合的头部(seq)
col + (ele,ele)
//将其他集合添加到集合尾部(set/map)
col -(ele,ele)
//将子集合从集合中删除(set/map/ArrayBuffer)

col1 ++ col2
//将其他集合添加到集合尾部(Iterator)

col2 ++: col1 
//将其他集合添加到集合的头部(Iterator)

ele::list
//将元素添加到list头部(list)
list2::list1
//将其他list2添加到list1
的头部(list)
list1:::list2 
//将其他list2添加到list1的尾部(list)

4.2 序列

序列分为Vector,Range,两个都是不可变的
Vector操作:

Vector是一个带下标的序列,可以通过下标(索引号),来访问Vector元素
scala> var v = Vector(1,2,3,4,5,6)
v: scala.collection.immutable.Vector[Int] = Vector(1, 2, 3, 4, 5, 6)

Range操作:

Range:是一个整数的序列

第一种写法:
scala> Range(0,5)
res48: scala.collection.immutable.Range = Range(0, 1, 2, 3, 4)
解释:从0开始,不包含5

第二种写法:
scala> print(0 until 5)
Range(0, 1, 2, 3, 4)

第三种写法:前后闭区间
scala> print(0 to 5)
Range(0, 1, 2, 3, 4, 5)

两个Range可以相加
scala> ('0' to '9') ++ ('A' to 'Z')
res51: scala.collection.immutable.IndexedSeq[Char] = Vector(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, A, B, C, D, E, F, G, H, I, J, K, L, M, N, O, P, Q, R, S, T, U, V, W, X, Y, Z)

Range转换成list
scala> 1 to 5 toList
warning: there was one feature warning; re-run with -feature for details
res52: List[Int] = List(1, 2, 3, 4, 5)

4.3 集合set

不重复元素的集合,默认是HashSet,与java类似,且不可变,相关操作

创建一个set
scala> var s1 = Set(1,2,10,8)
s1: scala.collection.immutable.Set[Int] = Set(1, 2, 10, 8)
注意:属于immutable

scala> s1 + 10
res53: scala.collection.immutable.Set[Int] = Set(1, 2, 10, 8)

scala> s1 + 7
res54: scala.collection.immutable.Set[Int] = Set(10, 1, 2, 7, 8)
返回一个新Set

scala> s1
res55: scala.collection.immutable.Set[Int] = Set(1, 2, 10, 8)
s1本身未改变

创建一个可排序Set
scala> var s2 = scala.collection.mutable.SortedSet(1,2,3,10,8)
s2: scala.collection.mutable.SortedSet[Int] = TreeSet(1, 2, 3, 8, 10)

判断元素是否存在:
scala> s1.contains(1)
res56: Boolean = true

判断一个集是否是另一个集的子集
scala> var s2 = Set(1,2,10,8,7,0)
s2: scala.collection.immutable.Set[Int] = Set(0, 10, 1, 2, 7, 8)

scala> s1
res57: scala.collection.immutable.Set[Int] = Set(1, 2, 10, 8)

scala> s1 subsetOf(s2)
res58: Boolean = true

集的运算:union 并集, intersect 交集,diff 差集
scala> var set1 = Set(1,2,3,4,5,6)
set1: scala.collection.immutable.Set[Int] = Set(5, 1, 6, 2, 3, 4)

scala> var set2 = Set(5,6,7,8,9,10)
set2: scala.collection.immutable.Set[Int] = Set(5, 10, 6, 9, 7, 8)

scala> set1 union set2
res59: scala.collection.immutable.Set[Int] = Set(5, 10, 1, 6, 9, 2, 7, 3, 8, 4)

scala> set1 intersect set2
res60: scala.collection.immutable.Set[Int] = Set(5, 6)

scala> set1 diff set2
res61: scala.collection.immutable.Set[Int] = Set(1, 2, 3, 4)

scala> set2 diff set1
res62: scala.collection.immutable.Set[Int] = Set(10, 9, 7, 8)

五、scala中的函数

前面说的到普通函数的定义方式为:

关键字 def 函数名(参数):返回值 = 函数实现

除了普通函数之外,还有其他函数。

5.1 匿名函数

没有名字的函数,定义方式为:

(参数列表)=> {函数实现}
注意,是 => 不是 =

一般来说,匿名函数就是用来临时定义一个函数的,只使用一次,常常和高阶函数结合使用。例子:

调用匿名函数
定义一个数组,把数组中每个元素乘以三
把Array(1, 2, 3)中的每个元素,传到匿名函数中(x:Int) => x*3

scala> Array(1,2,3).map((x:Int) => x*3)
res1: Array[Int] = Array(3, 6, 9)

把 (x:Int) => x*3 作为map的函数参数传入   ---> 高阶函数

(_,1)这个是啥意思  _+_  都是匿名函数。

 定义匿名函数的时候,参数列表可以省略,直接用函数体,如:
(i:int)=>i*2 可以省略为 _*2,其中_ 就表示参数,输出类型自动检测

要注意一点:
匿名函数的输入参数中,不强制需要指定参数的类型,可以自动检测,如:

xxx.map(pair=>pair._2.toString)
map里面的匿名函数是合法的,没有指定输入参数类型,也是OK的。

但是普通函数是一定要指定参数类型的。

5.2 高阶函数

高阶函数就是参数列表中使用另外的函数作为参数。使用方式:

定义一个高阶函数:
def 函数名(传入函数名,假设为f:(入参类型)=>(出参类型),参数2,。。。。) = 处理逻辑,这里面会调用传入的函数

例子:
object IODemo {

  def main(args: Array[String]): Unit = {
    //调用高阶函数,并传入函数作为参数
    highFun(innerFun,2)
  }

  //高阶函数,有两个参数,一个是函数f,指定入参和出参类型均为Int,另一个参数是x:Int
  def highFun(f:(Int)=>(Int),x:Int) = {
    f(x)
  }

  //这个是接着用来传入到高阶函数中的函数
  def innerFun(x:Int) = {
    x+2
  }

}

除了使用已有函数作为高阶函数的参数之外,也可以使用匿名函数,如:
highFun((x:Int)=>x*3, 3),(x:Int)=>x*3就是个匿名函数

5.3 scala中常用高阶函数

5.3.1 map函数

相当于一个循环,对某个集合中的每个元素进行操作(按照接收函数中的逻辑),返回一个处理后的数据的新的集合。例子:

scala> var numbers = List(1,2,3,4,5,6,7,8,9,10)
numbers: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> numbers.map((i:Int)=>i*2)
res8: List[Int] = List(2, 4, 6, 8, 10, 12, 14, 16, 18, 20)
在(i:Int)=>i*2中,i是循环变量,整个函数

map函数不改变numbers值
scala> numbers
res10: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

匿名函数可以简写为;
_ 相当于循环变量 i
_*2 与 (i:Int)=>i*2 功能相同
_+_ 与 (i:Int,j:Int)=>i+j

注意:如果传给map的是一个Map类型的,如果key没有给map中的函数处理,只处理value,那么最后map处理后返回的新的集合就只包含value

5.3.2 foreach函数

和map类似,唯一不同的就是它没有任何返回值

scala> numbers
res11: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> numbers.foreach(_*2)

scala> numbers
res13: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> numbers.foreach(println(_))
1
2
3
4
5
6
7
8
9
10

scala> numbers.map(_*2).foreach(println)
2
4
6
8
10
12
14
16
18
20

5.3.3 filter

过滤,选择满足的数据,返回true则保留,false则丢弃。最终返回的新集合中包含满足条件的元素。例子:

举例:查询能够被2整除的数字
scala> numbers
res15: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> numbers.filter((i:Int)=>i%2==0)
res16: List[Int] = List(2, 4, 6, 8, 10)

说明:(i:Int)=>i%2==0  如果是true 就返回

5.3.4 zip

合并两个集合,例子:

scala> List(1,2,3).zip(List(4,5,6))
res18: List[(Int, Int)] = List((1,4), (2,5), (3,6))

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

可以看到,他是两个集合中的元素两两集合成一个元祖,并且按照最少元素的list为准

scala> List(3).zip(List(4,5))
res20: List[(Int, Int)] = List((3,4))

5.3.5 partition

根据断言(就是某个条件,可以通过一个匿名函数来实现)的结果,进行分区。函数返回的类型必须是boolean类型的。最后也就是返回true的为一个分区,为false的为另一个分区。如:

把能够被2整除的分成一个区,不能整除的分成另一个区
scala> numbers
res21: List[Int] = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)

scala> numbers.partition((i:Int)=>i%2==0)
res22: (List[Int], List[Int]) = (List(2, 4, 6, 8, 10),List(1, 3, 5, 7, 9))

5.3.6 find

查找第一个满足条件(断言)的元素

举例:查找第一个能够被3整除的数字
scala> numbers.find(_%3==0)
res23: Option[Int] = Some(3)

5.3.7 flatten

把嵌套的结果展开

scala> List(List(2,4,6,8,10),List(1,3,5,7,9)).flatten
res24: List[Int] = List(2, 4, 6, 8, 10, 1, 3, 5, 7, 9)

合并成为一个集合

5.3.8 faltmap

相当于map+flatten

scala> var myList = List(List(2,4,6,8,10),List(1,3,5,7,9))
myList: List[List[Int]] = List(List(2, 4, 6, 8, 10), List(1, 3, 5, 7, 9))

scala> myList.flatMap(x=>x.map(_*2))
res25: List[Int] = List(4, 8, 12, 16, 20, 2, 6, 10, 14, 18)

执行过程:
1、将List(2,4,6,8,10)和List(1,3,5,7,9)循环调用了x=>x.map(_*2)  这里x代表某个List,里面的_表示list中的元素
List(4, 8, 12, 16, 20)  和 List(2, 6, 10, 14, 18)

2、合并成一个List
List(4, 8, 12, 16, 20, 2, 6, 10, 14, 18)

5.4 闭包

闭包也就是函数的嵌套。在一个函数里面,包含了另外一个函数的定义,可以在内函数中访问外函数的变量。如:

def mulBy(factor:Double)=(x:Double)=>x*factor里面调用了外函数的factor
            外                       内

5.5 柯里化

柯里化函数,是把具有多个参数的函数,转换成为一个函数链,每个节点上都是单一参数函数

def add(x:Int,y:Int)=x+y
def add(x:Int)(y:Int)=x+y
以上两个函数定义时等价的

说明:
普通函数:
def add(x:Int,y:Int)=x+y

柯里化函数:利用上面函数闭包的方式
def add(x:Int)=(y:Int)=>x+y

简写:
def add(x:Int)(y:Int)=x+y

scala> def add(x:Int)(y:Int)=x+y
add: (x: Int)(y: Int)Int

scala> add(1)(2)
res28: Int = 3

5.6 解答疑问

有时候调用方法时,方法名后面不需要加括号,有时候需要?为什么呢?

主要看定义函数的时候,后面有没有带括号,当然这是在函数没有参数的情况下,有参数肯定就有括号的。
当函数是无参函数,且定义时没有带括号,那么调用时必须不带括号
当函数是无参函数,且定义是带括号了,那么调用时括号可有可无

同样的情况在定义类的时候也是类似的,不重复

六、高阶特性

6.1 scala中特殊的类型

* Any 表示任意类型 相当于java中的Object
* Unit 表示无值,相当于 void
* Nothing Nothing类型在Scala的类层级的最低端;它是任何其他类型的子类型
* Null 是所有应用类型的子类,值为null
*
* 特殊的类型:
* Option : Scala Option(选项)类型用来表示一个值是可选的(有值或者无值)
* Some :如果值存在,Option 就是 Some
* None : 如果值不存在,Option 就是 None
*
* scala> var myMap = Map("Andy"->90)
myMap: scala.collection.immutable.Map[String,Int] = Map(Andy -> 90)

scala> myMap.get("Andy")
res0: Option[Int] = Some(90)

scala> myMap.get("ykjdfhsdajfkajshd")
res1: Option[Int] = None
*
* Nil类型:是一个空的List
*
* 四个N总结:None Nothing Null Nil
* None : 如果map中的值不存在,Option 就是 None
* Nothing : 如果方法抛出异常,返回值类型是Nothing,Nothing是任何其他类型的子类型
* Null : 可以赋值给所有引用类型,不能赋值给值类型。
* Nil : 空的List

6.2模式匹配

类似于java中的switch/case,用法:

xx match {
    case xx1=>。。。。  匹配1
    case xx2=>。。。。  匹配2
    case _ => ......    这里是默认值的意思,相当于default   
   }

例子:

object Demo1 {

  def main(args: Array[String]): Unit = {
    //1、相当于switch case
    var chi = '-'
    var sign = 0 //标识符  如果 chi 为 ‘-’ 则sign赋值为-1

    chi match {
      case '+' => sign = 1
      case '-' => sign = -1
      case _ => sign=0 // _ 表示其他的值
    }
    println(sign)

    /**
      * 2、Scala 的守卫:匹配某种类型的所有值 case _ if
      *
      * 举例 匹配所有的数字,如果ch2是一个数字,则digit赋值为ch2
      */

    var ch2 = '6'
    var digit : Int = -1

    ch2 match {
      case '+' => println("这是一个加号")
      case '-' => println("这是一个减号")
      case _ if Character.isDigit(ch2) => digit = Character.digit(ch2,10)// 10 表示 10进制
      case _ => println("其他")
    }

    println(digit)

    /**
      * 3、在模式匹配中使用变量
      */
    var mystr = "Hello World"
    //取出某个字符,赋给模式匹配的变量
    mystr(7) match {
      case '+' => println("这是一个加号")
      case '-' => println("这是一个减号")
      case ch => println(ch)//case语句中使用变量 ch代表传递进来的字符
    }

    /**
      * 4、匹配类型:相当于java中的instanceof
      */
    var v4 : Any = 100 // 最终 v4 是一个整数
    v4 match {
      case x : Int => println("这是一个整数")
      case s : String => println("这是一个字符串")
      case _ => println("其他类型")
    }

    /**
      * 5、匹配数组和列表
      */
    var myArray = Array(1,2,3)
    myArray match {
      case Array(0) => println("数组中只有一个0")
      case Array(x,y) => println("数组中包含两个元素")
      case Array(x,y,z) => println("数组中包含3个元素")
      case Array(x,_*) => println("这是一个数组,包含多个元素")
    }

    var myList = List(1,2,3,4,5,6)
    myList match {
      case List(0) => println("列表中只有一个0")
      case List(x,y) => println("列表中包含两个元素,和是: " + (x+y))
      case List(x,y,z) => println("列表中包含3个元素,和是: "+ (x+y+z))
      case List(x,_*) => println("这是一个列表,包含多个元素,和是: " + myList.sum)
    }

  }

}

6.3 样本类case

样本类比起普通的类,多了一个特点,可以用上面的case语句中,作为匹配类型,普通的类是不可以的。其他用法和普通类一致。定义方式:

case class a(x:int.....) {}

注意一点,样本类中,自动会将所有字段都声明为val,所以我们声明字段时可以省略val关键字

具体其实没什么好说的,用法很简单。常用来作为一些数据的存储类

6.4 泛型

scala中泛型的定义和java类似,这里不重复,直接讲用法。

6.4.1 泛型类

定义类的时候,带有一个泛型,如:

class Father[T]{
  xxxxx
}

定义的方式和java中的一样,scala中很多都带有泛型,比如:

Array[T]
List[T]
Map[K,V]

6.4.2 泛型函数

定义函数的时候,定义一个泛型,如:

def mkArray[T:ClassTag]
ClassTag解释:表示在scala在运行时候的状态信息,这里表示调用时候的数据类型

例子:

scala> import scala.reflect.ClassTag
import scala.reflect.ClassTag

定义了一个泛型数组,其中elem:_*表示所有元素
scala> def mkArray[T:ClassTag](elem:T*) = Array[T](elem:_*)
mkArray: [T](elem: T*)(implicit evidence$1: scala.reflect.ClassTag[T])Array[T]

6.5 隐式转换

6.5.1 隐式转换函数

通常意思是scala自动将某些类型转换为指定类型,只要用户定义相关的隐式转换函数,scala会自动根据隐式函数的入参和出参类型来调用。例子:

class Fruit(name:String){
  def getFruitName() : String = name
}

class Monkey(f:Fruit){
  //输出
  def say() = println("Monkey like " + f.getFruitName())
}

object ImplicitDemo {
  def main(args: Array[String]): Unit = {
    //定义一个水果对象
    var f : Fruit = new Fruit("Banana")

    f.say()
    /**
      * 问题:可否 f.say
      * 直接写会报错,因为 Fruit里面没有say函数
      * 但是 Monkey里面有say函数
      *
      * 如果可以把 Fruit 转换成 Monkey的话,就可以调用say
      *
      * 所以 我们定义隐式转换函数
      *
      */
  }

  //定义隐式转换函数
  implicit def fruit2Monkey(f:Fruit):Monkey = {
    new Monkey(f)
  }

  /**
    * 注意:谨慎使用隐式转换,隐式转换会导致scala可读性进一步变差
    *
    * 隐式转换函数命名:xxx2xxx
    */
}

6.5.2 隐式参数

隐式参数就是在函数定义时,在每个参数前面加上一个 implicit关键字,该参数就是隐式参数。例子:

定义一个带有隐式参数的函数
scala> def testParam(implicit name:String) = println("The value is " + name)
testParam: (implicit name: String)Unit

接着定义一个隐式变量
scala> implicit val name:String = "AAAAAA"
name: String = AAAAAA

传入参数调用方法,可以运行的
scala> testParam("dfsfdsdf")
The value is dfsfdsdf

没有传递参数时,自动寻找已定义的隐式变量,就是前面的name,当前,调用隐式变量时并不是按照名字调用的
scala> testParam
The value is AAAAAA

有多个隐式变量时,如果同类型的有多个,会报错。如果是不同类型就可以
scala> implicit val name2:String = "AAAAAACCCCCCCCCCCCCC"
name2: String = AAAAAACCCCCCCCCCCCCC
scala> testParam
:18: error: ambiguous implicit values:
both value name of type => String
and value name2 of type => String
match expected type String
testParam

注意:定义隐式参数时,只能有一个隐式参数,而且有时候为了便于看,会将隐式参数和普通参数分开定义,如:
def test1(c:Int)(implicit a:Int): Unit = {}
注意,普通参数一定要在前面的括号,不能放在后面

还有一个特别的例子,将隐式参数和隐式转换函数结合:

定义一个隐式参数,实现如下需求:找到两个值中比较小的那个值
100 23 ----> 23
"Hello" "ABC"--->ABC

def smaller[T](a:T,b:T)(implicit order : T => Ordered[T]) = if(a def smaller[T](a:T,b:T)(implicit order : T => Ordered[T]) = if(a Ordered[T])T

scala> smaller(100,23)
res1: Int = 23

scala> smaller("Hello","ABC")
res2: String = ABC

首先泛型T,我们不确定是否能比较的,也就是里面有没有 < 这个方法(不要奇怪,< 是一个方法名,不是符号),因为我们需要确定是否有,则我们定义一个隐式转换函数和参数:
implicit order : T => Ordered[T]  
首先这是一个匿名的隐式转换函数,将T转为Order[T],并作为参数order的值

6.5.3 隐式类

在类名前加 implicit关键字
作用:通过隐式类来增强某个类的功能,会将某个类的对象转换为隐式类对象,然后就可调用隐式类中定义的方法了。例子:

object ImplicitClassDemo {

  def main(args: Array[String]): Unit = {
    //执行两个数字求和
    println("两个数字的和是: " + 1.add(2)) // 使用 1.add(2)  替换 1+2
    /**
      * 1.add(2) 报错 因为 1 没有 add方法
      *
      * 定义一个隐式类来增强 1 的功能
      *
      * 执行过程:
      * 首先 把 1 转换成 Calc(1)
      * 再调用Calc(1) add方法
        是根据隐式类中接收的参数类型来自动调用的
      */

    implicit class Calc(x:Int){
      def add(y:Int):Int = x + y
    }
  }

}

6.6 泛型的上界和下界

有时候想限制定义的泛型只能是某些类的时候,就会用到上下界的概念。
例子:

(*)规定泛型的取值范围
举例:定义一个泛型:T
类的继承关系   A---->B----->C----->D  箭头指向子类
可以规定T的取值范围    D  <:  T   <:  B
T 的取值范围只能是 B C D

<:  就是上下界表示方法

(*)定义:
上界  s <: T 规定了S的类型必须是T的子类或本身
下界  u >: T 规定了U的类型必须是T的父类或本身

关于下界的一个小问题:

class Father
class Son extends Father
class Son1 extends Son
class Son2 extends Son1

object Demo2 {
  def main(args: Array[String]): Unit = {

    var father : Father = new Son2
    fun2(father)

    //fun2(new Son2)  这个能正常运行,为什么?
    主要是因为子类的对象可以赋给父类的引用,类似于自动转型。所以new son2的时候,实际上是可以当做是父类的对象的,所以可以正常运行。由此可以看出,在继承的情况下其实泛型的下界是没什么用的。其实也就是多态的原因
  }

  def fun[T<:Son](x:T)={
    println("123")
  }

  def fun2[T>:Son](x:T): Unit ={
    println("456")
  }

}

6.7 视图界定

上界和下界的一种扩展,除了可以接收上界和下界规定的类型之外,还可以接收能够通过隐式转换过去的类型。如:

用法 :
def addTwoString[T<%String](x:T,y:T) = x + "  " + y
含义:
1、可以接收String及其子类
2、可以接收能够转换成String的其他类型。这个是重点,也就是隐式转换来的

例子:

scala> def addTwoString[T<%String](x:T,y:T) = x + " **** " + y
addTwoString: [T](x: T, y: T)(implicit evidence$1: T => String)String

scala> addTwoString(1,2)
:14: error: No implicit view available from Int => String.
addTwoString(1,2)

报错解决:定义隐式转换函数,把int转换成String

scala> implicit def int2String(n:Int):String=n.toString
warning: there was one feature warning; re-run with -feature for details
int2String: (n: Int)String

scala> addTwoString(1,2)
res13: String = 1 **** 2

分析执行过程:
1、调用 int2String 方法,把Int转换成String(scala在后台调用,不需要显示调用)
2、调用addTwoString方法,拼接字符串

隐式转换函数,不需要显示调用它,它就能被调用。

6.8 协变和逆变

将泛型变量的类型嫁接到泛型类中

概念:
协变:泛型变量的值可以是本身类型或者其子类型,
逆变:泛型变量的值可以是本身类型或者其父类型

表示: 
协变 +  class A[+T] 表示将T泛型的类型嫁接到A中
逆变 -

6.9 Ordered和Ordering比较

这两个类主要用于比较时使用
Ordered类似于java的comparable接口
Ordering类似于java的compartor接口。

trait Ordered[A] extends scala.Any with java.lang.Comparable[A] {
  def compare(that : A) : scala.Int
  def <(that : A) : scala.Boolean = { /* compiled code */ }
  def >(that : A) : scala.Boolean = { /* compiled code */ }
  def <=(that : A) : scala.Boolean = { /* compiled code */ }
  def >=(that : A) : scala.Boolean = { /* compiled code */ }
  def compareTo(that : A) : scala.Int = { /* compiled code */ }
}

这其实就是一个trait,定义了< 等方法,不用奇怪,这就是方法名。
子类可以继承该trait,重写里面的方法,该类就可以用于比较了。

trait Ordering[T] extends java.lang.Object with java.util.Comparator[T] with scala.math.PartialOrdering[T] with scala.Serializable {
 this : scala.math.Ordering[T] =>
  def tryCompare(x : T, y : T) : scala.Some[scala.Int] = { /* compiled code */ }
  def compare(x : T, y : T) : scala.Int
  override def lteq(x : T, y : T) : scala.Boolean = { /* compiled code */ }
  override def gteq(x : T, y : T) : scala.Boolean = { /* compiled code */ }
  override def lt(x : T, y : T) : scala.Boolean = { /* compiled code */ }
  override def gt(x : T, y : T) : scala.Boolean = { /* compiled code */ }
  override def equiv(x : T, y : T) : scala.Boolean = { /* compiled code */ }
  def max(x : T, y : T) : T = { /* compiled code */ }
  def min(x : T, y : T) : T = { /* compiled code */ }
  override def reverse : scala.math.Ordering[T] = { /* compiled code */ }
  def on[U](f : scala.Function1[U, T]) : scala.math.Ordering[U] = { /* compiled code */ }
  class Ops(lhs : T) extends scala.AnyRef {
    def <(rhs : T) : scala.Boolean = { /* compiled code */ }
    def <=(rhs : T) : scala.Boolean = { /* compiled code */ }
    def >(rhs : T) : scala.Boolean = { /* compiled code */ }
    def >=(rhs : T) : scala.Boolean = { /* compiled code */ }
    def equiv(rhs : T) : scala.Boolean = { /* compiled code */ }
    def max(rhs : T) : T = { /* compiled code */ }
    def min(rhs : T) : T = { /* compiled code */ }
  }

ordering的方法就比较多,用法类似