泛类型
和java一样,scala也有泛型的概念,在scala里面泛型是使用方括号 [] 来接受类型参数的。通常使用字母A来作为参数标志符,当然你也可以使用其他任意的参数名称。
class Stack[A] {private var elements: List[A] = Nildef push(x: A) { elements = x :: elements }def peek: A = elements.headdef pop(): A = {val currentTop = peekelements = elements.tailcurrentTop}}
要使用一个泛类型,将一个具体的类型替换掉A即可。
val stack = new Stack[Int]stack.push(1)stack.push(2)println(stack.pop) // prints 2println(stack.pop) // prints 1
上面的例子中,实例对象接收整型值,如果该类型有子类型,子类型也是可以传入的。
class Fruitclass Apple extends Fruitclass Banana extends Fruitval stack = new Stack[Fruit]val apple = new Appleval banana = new Bananastack.push(apple)stack.push(banana)
变和逆变
1.语法
class MyList[+T]{ //协变}class MyList[-T]{ //逆变}class MyList[T] //不变
2.说明
协变:Son是Father的子类,则MyList[Son] 也作为MyList[Father]的“子类”。
逆变:Son是Father的子类,则MyList[Son]作为MyList[Father]的“父类”。
不变:Son是Father的子类,则MyList[Father]与MyList[Son]“无父子关系”。
3.案例实操
//泛型模板//class MyList<T>{}//不变//class MyList[T]{}//协变//class MyList[+T]{}//逆变//class MyList[-T]{}// 定义继承关系class Parent {}class Child extends Parent {}class SubChild extends Child {}// 定义带泛型的集合类型, 定义不同的逆变协边, 可以让 =号前后, 用父子类关系class MyCollection[-E] {}object Scala_TestGeneric {def main(args: Array[String]): Unit = {// 1. 协变和逆变val child: Parent = new Child// val childList: MyCollection[Parent] = new MyCollection[Child]val childList: MyCollection[SubChild] = new MyCollection[Child]// 2. 上下限def test[A <: Child](a: A): Unit = {println(a.getClass.getName)}test[SubChild](new SubChild)}}
泛型上下限
1.语法
Class PersonList[T <: Person]{ //泛型上限, 传的类型可以是Person和他的子类}Class PersonList[T >: Person]{ //泛型下限, 传的类型可以是Person和他的父类}
2.说明
泛型的上下限的作用是对传入的泛型进行限定
3.案例实操
class Parent {}class Child extends Parent {}class SubChild extends Child {}object Scala_TestGeneric {def main(args: Array[String]): Unit = {//test(classOf[SubChild])//test[Child](new SubChild)}//泛型通配符之上限//def test[A <: Child](a:Class[A]): Unit ={// println(a)//}//泛型通配符之下限//def test[A >: Child](a:Class[A]): Unit ={// println(a)//}//泛型通配符之下限 形式扩展def test[A >: Child](a: A): Unit = {println(a.getClass.getName)}}
上下文限定
1.语法
//A是一个泛型, B是隐式参数的泛型名称而B这个泛型的名称里面的泛型又是A, 用这个隐式参数用implicitlydef f[A : B](a: A) = println(a)//等同于def f[A](a:A)(implicit arg:B[A])=println(a)
2.说明
上下文限定是将泛型和隐式转换的结合产物,以下两者功能相同,使用上下文限定 [A : Ordering] 之后,方法内无法使用隐式参数名调用隐式参数,需要通过implicitly[Ordering[A]] 获取隐式变量,如果此时无法查找到对应类型的隐式变量,会发生出错误
implicit val x = 1val y = implicitly[Int]val z = implicitly[Double]
3.案例实操
def f[A:Ordering](a:A,b:A) =implicitly[Ordering[A]].compare(a,b)def f[A](a: A, b: A)(implicit ord: Ordering[A]) = ord.compare(a, b)
