scala 系列之 18scala 泛型

教程 潘牛 ⋅ 于 2021-07-08 22:56:06 ⋅ 1995 阅读

22 泛型

泛型就是不确定的类型,可以在类或方法不确实传入类型时使用,可以提高代码的灵活性和复用性;

scala中泛型的用法和java中差不多,但是会有一些自己独特的语法;

泛型类:指定类可以接受任意类型参数。

泛型方法:指定方法可以接受任意类型参数。

22.1 泛型类基本用法

package day04
import day04.SexEnumObj.SexEnum
// 定义带有泛型的抽象类
abstract class FXDemo[T](val t : T) {
  def printInfo():Unit
}
// 子类继承父类,把泛型具体化成Int
class FXIntDemo[Int](t : Int) extends FXDemo[Int](t:Int){
  override def printInfo(): Unit = {
    println(s"FXIntDemo[Int](${t})")
  }
}
// 子类继承父类,把泛型具体化成String
class FXStringDemo[String](t : String) extends FXDemo[String](t:String){
  override def printInfo(): Unit = {
    println(s"FXIntDemo[String](${t})")
  }
}
// 定义带有多泛型的类
class FXABCDemo[A, B, C](val a:A, val b:B, val c:C){
  override def toString: String = s"${a}, ${b}, ${c}"
}
// 定义sex的枚举对象
object SexEnumObj extends Enumeration{
  // 定义枚举类型(用于泛型)
  type SexEnum = Value
  // 定义枚举值
  val boy, girl = Value
}
object FXDemo{
  def main(args: Array[String]): Unit = {
    val demo = new FXIntDemo[Int](100)
    demo.printInfo()
    val demo2 = new FXStringDemo[String]("xiaoming")
    demo2.printInfo()
    val demo3 = new FXABCDemo[String, Int, String]("xiaoming", 20, "boy")
    println(demo3)
    val demo4 = new FXABCDemo[String, Int, SexEnum]("xiaoming", 20, SexEnumObj.boy)
    println(demo4)
  }
}

22.2 泛型种类

[B <: A] UpperBound 上界,B类型的父类是A类型,左侧的B的类型必须是A类型或者A类型的子类。

[B >: A] LowerBound 下界,B类型的子类是A类型,左侧的B的类型必须是A类型或者A类型的父类。

[B <% A] ViewBound B类型转换成A类型,需要一个隐式转换函数。

[B : A] ContextBound 通过隐式转换增加个 A[B] 类型。

[-A] 逆变,通常作为参数类型,T是A的子类。

[+B] 协变,通常作为返回类型,T是B的父类。

22.2.1 UpperBound

UpperBound 用在泛型类或泛型方法上,用于限制传递的参数必须是 指定类型对象或其子类对象。

如果想实现两个对象的比较,需要该对象实现Comparable接口。然后再配上泛型实现通用比较。

泛型继承,java的用法

package javaday04;
public class UpperBoundDemo<T extends Comparable<T>> {
    // psvm
    public static void main(String[] args) {
        // Integer 实现了 Comparable接口,创建对象时,约束通过
        UpperBoundDemo<Integer> demo1 = new UpperBoundDemo<Integer>();
        // Hainiu 没实现 Comparable接口,创建对象时,约束不通过
//        UpperBoundDemo<Hainiu> demo2 = new UpperBoundDemo<Hainiu>();
        // 约束通过
        UpperBoundDemo<HainiuComparable> demo3 = new UpperBoundDemo<HainiuComparable>();
    }
}
class Hainiu{}
class HainiuComparable implements Comparable<HainiuComparable>{
    public int compareTo(HainiuComparable o) {
        return 0;
    }
}

泛型继承,scala用法

package day04
// 在类上定义泛型, new对象时就会约束
class UpperBoundDemo[T <: Comparable[T]](val t:T) {
}

object UpperBoundDemo{
  def main(args: Array[String]): Unit = {
    // 因为 Integer 是 Comparable 实现类,所以约束通过
    val demo = new UpperBoundDemo[Integer](100)
    println(demo.t)

    // 约束通过
    val demo2 = new UpperBoundDemo[HainiuComparable](new HainiuComparable())
  }
}

class HainiuComparable implements Comparable<HainiuComparable>{
    public int compareTo(HainiuComparable o) {
        return 0;
    }
}

22.2.2 LowerBound

LowerBound 用在泛型类或泛型方法上,用于限制传递的参数必须是 指定类型对象或其父类对象。

package day04
class LowerBoundDemo[T] {
  // 将lowerbound约束到方法上
  def say[R>:T](r:R) = println(s"say ${r}")
}
object LowerBoundDemo{
  def main(args: Hainiurray[String]): Unit = {
    // new对象时指定泛型是Hainiu类型
    val demo = new LowerBoundDemo[Hainiu]
    // say方法是lowerbound约束, 只有Hainiu或Hainiu的父类可以通过约束
    demo.say[Hainiu](new Hainiu)
    demo.say[HainiuSupper](new HainiuSupper)
    // Hainiu的子类是不能通过约束的
//    demo.say[HainiuSub](new HainiuSub)
    // 如果调用方法时不加泛型,那就约束通过(不约束)
    demo.say(new HainiuSub)
  }
}
class HainiuSupper
class Hainiu extends HainiuSupper
class HainiuSub extends Hainiu

22.2.3 ViewBound [B <% A]

ViewBound 用在泛型类或泛型方法上,通过隐式转换把 B类型转换成A类型。

示例:

已知类 HainiuWork,对HainiuWork的两个对象作比较。

class HainiuWork(val company: String, val money: Int, val holiday: Int) {
  override def toString: String = {
    s"company:${company}, money:${money}, holiday:${holiday}"
  }
}

java的实现方法:定义外部比较器,用外部比较器去实现。

scala 用 viewbound 实现方法:通过隐式转换 将 HainiuWork 转成 Ordered[HainiuWork],然后通过Ordered 实现比较。

Ordered 是什么?

Ordered特质更像是rich版的Comparable接口,除了compare方法外,更丰富了比较操作(<, >, <=, >=)。

trait Ordered[A] extends Any with java.lang.Comparable[A] {
    def compare(that: A): Int    // 内部比较器的比较方法
    def <  (that: A): Boolean = (this compare that) <  0
    def >  (that: A): Boolean = (this compare that) >  0
    def <= (that: A): Boolean = (this compare that) <= 0
    def >= (that: A): Boolean = (this compare that) >= 0
    def compareTo(that: A): Int = compare(that)   // 外部比较器的比较方法

比较代码实现:

// Ordered 支持 > 操作
class ViewBoundDemo[W <% Ordered[W]] {
  def choose(work1: W, work2: W): W = {
    if (work1 > work2) work1 else work2
  }
}
object ViewBoundDemo {
  def main(args: Array[String]): Unit = {
    import MyPredef.selectWork
    val demo = new ViewBoundDemo[HainiuWork]

    val work1 = new HainiuWork("金融", 20000, 16)
    val work2 = new HainiuWork("互联网", 20000, -6)

    // 通过隐式转换将 HainiuWork --> Ordered[HainiuWork]
    println(demo.choose(work1, work2))
  }
}
class HainiuWork(val company: String, val money: Int, val holiday: Int) {
  override def toString: String = {
    s"go to ${company},happy holiday ${holiday}"
  }
}

隐式转换函数实现

  implicit val selectWork = (s:HainiuWork) => new Ordered[HainiuWork]{
    override def compare(that: HainiuWork) = {
      // 先比较工资,工资相同比较假期
      if(s.money == that.money){
        s.holiday - that.holiday
      }else{
        s.money - that.money
      }
    }
  }

22.2.4 ContextBound [B : A]

ContextBound 用在泛型类或泛型方法上,通过隐式转换给B类型增加个 隐式值A[B]类型。

还是对HainiuWork的两个对象作比较。

scala 用 ContextBound 实现方法: 通过隐式转换 将 HainiuWork 转成 Ordering[HainiuWork] 。

file

实现逻辑:

package day04
class ContextBoundDemo[T] {
  // 带有隐式参数的方法,那调用时,当前环境得有Ordering[T] 类型的隐式对象
  def chooseBigger(w1:T, w2:T)(implicit ord:Ordering[T]):T = {
    if(ord.gt(w1, w2)) w1 else w2
  }
}
object ContextBoundDemo{
  def main(args: Array[String]): Unit = {
    val w1 = new HainiuWork("互联网1", 30000, -3)
    val w2 = new HainiuWork("互联网2", 20000, 20)
    val demo = new ContextBoundDemo[HainiuWork]
    // 在调用前,引入隐式对象
    import util.MyPredef.HWOrdering
    val bigger = demo.chooseBigger(w1, w2)
    println(s"更幸福的是:${bigger}")
  }
}

升级版:

package day04
// 当new对象时,通过隐式转换会生成隐式的Ordering[T]
class ContextBoundDemo2[T : Ordering] {
  // 带有隐式参数的方法,那调用时,当前环境得有Ordering[T] 类型的隐式对象
  def chooseBigger(w1:T, w2:T):T = {
    // 方法1:
//    val ord = implicitly[Ordering[T]]
    // 方法2:
//    val ord = Ordering[T]
//    if(ord.gt(w1, w2)) w1 else w2
    // 方法3:
    import Ordered.orderingToOrdered
    if(w1 > w2) w1 else w2
  }
}
object ContextBoundDemo2{
  def main(args: Array[String]): Unit = {
    val w1 = new HainiuWork("互联网1", 30000, -3)
    val w2 = new HainiuWork("互联网2", 20000, 20)
    import util.MyPredef.HWOrdering
    val demo = new ContextBoundDemo2[HainiuWork]
    val bigger = demo.chooseBigger(w1, w2)
    println(s"更幸福的是:${bigger}")
  }
}

22.2.5 协变与逆变

在声明Scala的泛型类型时,“+”表示协变(covariance),而“-”表示逆变(contravariance)。

C[+T]:如果A是B的子类,那么C[A]是C[B]的子类;通常作为返回值类型。

C[-T]:如果A是B的子类,那么C[B]是C[A]的子类;通常作为参数类型。

C[T]:无论A和B是什么关系,C[A]和C[B]没有从属关系。

file

一切围绕着子类转父类可以直接转,但父类不能直接转子类,也就是泛型约束。

scala> class SuperA { def msuper() = println("SuperA") }
defined class SuperA
scala> class A extends SuperA { def m() = println("A") }
defined class A
scala> class SubA extends A { def msub() = println("SubA") }
defined class SubA
scala>
// 定义函数 入参是 A类型, 返回值是 A类型
scala> var func:(A)=>A = (a:A)=>a
func: A => A = <function1>
// 测试逆变
// function1 入参是逆变, 返回值是协变
// SuperA > A > SubA   ---> 逆变[SuperA] < 逆变[A] < 逆变[SubA]
// 函数的入参是SuperA,
scala> func = (a:SuperA)=> new A
// 调用时,将A对象传入,可以调用A的方法,SuperA的方法 
scala> func(new A).m
A
scala> func(new A).msuper
SuperA
func: A => A = <function1>
// 因为  逆变[A] < 逆变[SubA], 父类不能直接转子类, 约束不通过
scala> func = (a:SubA)=> new A
<console>:15: error: type mismatch;
 found   : SubA => A
 required: A => A
       func = (a:SubA)=> new A
                      ^
// 测试协变
// SuperA > A > SubA   ---> 协变[SuperA] > 协变[A] > 协变[SubA]
scala> func = (a:A) => new A
func: A => A = <function1>
scala> func = (a:A) => new SubA
func: A => A = <function1>
// 因为  协变[A] < 协变[SuperA], 父类不能直接转子类, 约束不通过
scala> func = (a:A) => new SuperA
<console>:14: error: type mismatch;
 found   : SuperA
 required: A
       func = (a:A) => new SuperA

示例:

// 协变
class T1[+T](e:T)
val v1:T1[java.lang.Integer] = new T1(100)
val v2:T1[java.lang.Integer] = v1
// 因为 Integer extends Number,所以 T1[Integer] 可以转换成 T1[Number], 这就是协变
val v3:T1[java.lang.Number] = v1 // 合法
// 反之不可以
val v4:T1[java.lang.Integer] = v3 //非法
// 逆变
class T2[-T](e:T)
val v1:T2[java.lang.Number] = new T2(100)
val v2:T2[java.lang.Number] = v1
// 因为Integer extends Number,所以 T1[Number] 可以转换成T1[Integer],这就是逆变
val v3:T2[java.lang.Integer] = v1 // 合法
// 反之不可以
val v4:T2[java.lang.Number] = v3 // 非法
版权声明:原创作品,允许转载,转载时务必以超链接的形式表明出处和作者信息。否则将追究法律责任。来自海汼部落-潘牛,http://hainiubl.com/topics/75753
成为第一个点赞的人吧 :bowtie:
回复数量: 0
    暂无评论~~
    • 请注意单词拼写,以及中英文排版,参考此页
    • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`, 更多语法请见这里 Markdown 语法
    • 支持表情,可用Emoji的自动补全, 在输入的时候只需要 ":" 就可以自动提示了 :metal: :point_right: 表情列表 :star: :sparkles:
    • 上传图片, 支持拖拽和剪切板黏贴上传, 格式限制 - jpg, png, gif,教程
    • 发布框支持本地存储功能,会在内容变更时保存,「提交」按钮点击时清空
    Ctrl+Enter