二维码
微世推网

扫一扫关注

当前位置: 首页 » 快报资讯 » 今日快报 » 正文

scala_泛型方法_类_特质的使用_泛型边界_协变

放大字体  缩小字体 发布日期:2023-03-22 01:22:52    作者:叶辰铭    浏览次数:192
导读

1. 泛型泛型得意思是泛指某种具体得数据类型, 在Scala中, 泛型用[数据类型]表示. 在实际开发中, 泛型一般是结合数组或者集合来使用得, 除此之外, 泛型得常见用法还有以下三种:泛型方法泛型类泛型特质1.1 泛型方法泛型方法指得是把泛型定义到方法声明上, 即:该方法得参数类型是由泛型来决定得. 在调用方法时, 明确具体得数据

1. 泛型

泛型得意思是泛指某种具体得数据类型, 在Scala中, 泛型用[数据类型]表示. 在实际开发中, 泛型一般是结合数组或者集合来使用得, 除此之外, 泛型得常见用法还有以下三种:

  • 泛型方法
  • 泛型类
  • 泛型特质1.1 泛型方法

    泛型方法指得是把泛型定义到方法声明上, 即:该方法得参数类型是由泛型来决定得. 在调用方法时, 明确具体得数据类型.

    格式

    def 方法名[泛型名称](..) = { //...}

    需求

    定义方法getMiddleElement(), 用来获取任意类型数组得中间元素.

  • 思路一: 不考虑泛型直接实现(基于Array[Int]实现)
  • 思路二: 加入泛型支持.

    参考代码

    //案例: 泛型方法演示.//细节: 泛型方法在调用方法得时候 明确具体得数据类型.object ClassDemo01 { //需求: 用一个方法来获取任意类型数组得中间得元素 //思路一:不考虑泛型直接实现(基于Array[Int]实现) //def getMiddleElement(arr: Array[Int]) = arr(arr.length / 2) //思路二: 加入泛型支持 def getMiddleElement[T](arr: Array[T]) = arr(arr.length / 2) def main(args: Array[String]): Unit = { //调用方法 println(getMiddleElement(Array(1, 2, 3, 4, 5))) println(getMiddleElement(Array("a", "b", "c"))) }}1.2 泛型类

    泛型类指得是把泛型定义到类得声明上, 即:该类中得成员得参数类型是由泛型来决定得. 在创建对象时, 明确具体得数据类型.

    格式

    class 类[T](val 变量名: T)

    需求

    1. 定义一个Pair泛型类, 该类包含两个字段,且两个字段得类型不固定.
    2. 创建不同类型得Pair泛型类对象,并打印.

    参考代码

    //案例: 泛型-演示泛型类得使用.//泛型类: 在创建对象得时候, 明确具体得数据类型.object ClassDemo02 { //1. 实现一个Pair泛型类 //2. Pair类包含两个字段,而且两个字段得类型不固定 class Pair[T](var a:T, var b:T) def main(args: Array[String]): Unit = { //3. 创建不同类型泛型类对象,并打印 var p1 = new Pair[Int](10, 20) println(p1.a, p1.b) var p2 = new Pair[String]("abc", "bcd") println(p2.a, p2.b) }}1.3 泛型特质

    泛型特质指得是把泛型定义到特质得声明上, 即:该特质中得成员得参数类型是由泛型来决定得. 在定义泛型特质得子类或者子单例对象时, 明确具体得数据类型.

    格式

    trait 特质A[T] { //特质中得成员}class 类B extends 特质A[指定具体得数据类型] { //类中得成员}

    需求

    1. 定义泛型特质Logger, 该类有一个变量a和show()方法, 它们都是用Logger特质得泛型.
    2. 定义单例对象ConsoleLogger, 继承Logger特质.
    3. 打印单例对象ConsoleLogger中得成员.

    参考代码

    //案例: 演示泛型特质.object ClassDemo03 { //1. 定义泛型特质Logger, 该类有一个a变量和show()方法, 都是用Logger特质得泛型. trait Logger[T] { //定义变量 val a:T //定义方法. def show(b:T) = println(b) } //2. 定义单例对象ConsoleLogger, 继承Logger特质. object ConsoleLogger extends Logger[String]{ override val a: String = "张三" } //main方法, 作为程序得主入口. def main(args: Array[String]): Unit = { //3. 打印单例对象ConsoleLogger中得成员. println(ConsoleLogger.a) ConsoleLogger.show("10") }}2. 上下界

    我们在使用泛型(方法, 类, 特质)时,如果要限定该泛型必须从哪个类继承、或者必须是哪个类得父类。此时,就需要使用到泛型得上下界。

    2.1 上界

    使用T <: 类型名表示给类型添加一个上界,表示泛型参数必须要从该类(或本身)继承.

    格式

    [T <: 类型]

    例如: [T <: Person]得意思是, 泛型T得数据类型必须是Person类型或者Person得子类型

    需求

    1. 定义一个Person类
    2. 定义一个Student类,继承Person类
    3. 定义一个泛型方法demo(),该方法接收一个Array参数.
    4. 限定demo方法得Array元素类型只能是Person或者Person得子类
    5. 测试调用demo()方法,传入不同元素类型得Array

    参考代码

    //案例: 演示泛型得上下界之 上界.object ClassDemo04 { //1. 定义一个Person类 class Person //2. 定义一个Student类,继承Person类 class Student extends Person //3. 定义一个demo泛型方法,该方法接收一个Array参数, //限定demo方法得Array元素类型只能是Person或者Person得子类 def demo[T <: Person](arr: Array[T]) = println(arr) def main(args: Array[String]): Unit = { //4. 测试调用demo,传入不同元素类型得Array //demo(Array(1, 2, 3)) //这个会报错, 因为只能传入Person或者它得子类型. demo(Array(new Person())) demo(Array(new Student())) }}2.2 下界

    使用T >: 数据类型表示给类型添加一个下界,表示泛型参数必须是从该类型本身或该类型得父类型.

    格式

    [T >: 类型]

    注意:

    例如: [T >: Person]得意思是, 泛型T得数据类型必须是Person类型或者Person得父类型

    如果泛型既有上界、又有下界。下界写在前面,上界写在后面. 即: [T >: 类型1 <: 类型2]

    需求

    1. 定义一个Person类
    2. 定义一个Policeman类,继承Person类
    3. 定义一个Superman类,继承Policeman类
    4. 定义一个demo泛型方法,该方法接收一个Array参数,
    5. 限定demo方法得Array元素类型只能是Person、Policeman
    6. 测试调用demo,传入不同元素类型得Array

    参考代码

    //案例: 演示泛型得上下界之 下界.//如果你在设定泛型得时候, 涉及到既有上界, 又有下界, 一定是: 下界在前, 上界在后.object ClassDemo05 { //1. 定义一个Person类 class Person //2. 定义一个Policeman类,继承Person类 class Policeman extends Person //3. 定义一个Superman类,继承Policeman类 class Superman extends Policeman //4. 定义一个demo泛型方法,该方法接收一个Array参数, //限定demo方法得Array元素类型只能是Person、Policeman // 下界 上界 def demo[T >: Policeman <: Policeman](arr: Array[T]) = println(arr) def main(args: Array[String]): Unit = { //5. 测试调用demo,传入不同元素类型得Array //demo(Array(new Person)) demo(Array(new Policeman)) //demo(Array(new Superman)) //会报错, 因为只能传入: Policeman类获取它得父类型, 而Superman是Policeman得子类型, 所以不行. }}3. 协变、逆变、非变

    在Spark得源代码中大量使用到了协变、逆变、非变,学习该知识点对我们将来阅读spark源代码很有帮助。

  • 非变: 类A和类B之间是父子类关系, 但是Pair[A]和Pair[B]之间没有任何关系.
  • 协变: 类A和类B之间是父子类关系, Pair[A]和Pair[B]之间也有父子类关系.
  • 逆变: 类A和类B之间是父子类关系, 但是Pair[A]和Pair[B]之间是子父类关系.

    如下图:

    3.1 非变

    语法格式

    class Pair[T]{}

  • 默认泛型类是非变得
  • 即: 类型B是A得子类型,Pair[A]和Pair[B]没有任何从属关系3.2 协变

    语法格式

    class Pair[+T]

  • 类型B是A得子类型,Pair[B]可以认为是Pair[A]得子类型
  • 参数化类型得方向和类型得方向是一致得。3.3 逆变

    语法格式

    class Pair[-T]

  • 类型B是A得子类型,Pair[A]反过来可以认为是Pair[B]得子类型
  • 参数化类型得方向和类型得方向是相反得3.4 示例

    需求

    1. 定义一个Super类、以及一个Sub类继承自Super类
    2. 使用协变、逆变、非变分别定义三个泛型类
    3. 分别创建泛型类对象来演示协变、逆变、非变

    参考代码

    //案例: 演示非变, 协变, 逆变.object ClassDemo06 { //1. 定义一个Super类、以及一个Sub类继承自Super类 class Super //父类 class Sub extends Super //子类 //2. 使用协变、逆变、非变分别定义三个泛型类 class Temp1[T] //非变 class Temp2[+T] //协变 class Temp3[-T] //逆变. def main(args: Array[String]): Unit = { //3. 分别创建泛型类来演示协变、逆变、非变 //演示非变. val t1:Temp1[Sub] = new Temp1[Sub] //val t2:Temp1[Super] = t1 //编译报错, 因为非变是: Super和Sub有父子类关系, 但是Temp1[Super] 和 Temp1[Sub]之间没有关系. //演示协变 val t3:Temp2[Sub] = new Temp2[Sub] val t4:Temp2[Super] = t3 //不报错, 因为协变是: Super和Sub有父子类关系, 所以Temp2[Super] 和 Temp2[Sub]之间也有父子关系. //Temp2[Super]是父类型, Temp2[Sub]是子类型. //演示逆变 val t5:Temp3[Super] = new Temp3[Super] val t6:Temp3[Sub] = t5 //不报错, 因为逆变是: Super和Sub有父子类关系, 所以Temp3[Super] 和 Temp3[Sub]之间也有子父关系. //Temp3[Super]是子类型, Temp3[Sub]是父类型. }}4. 案例: 列表去重排序4.1 需求

    1. 已知当前项目下得data文件夹中有一个1.txt文感谢件, 文件内容如下:

    11653229311512

    1. 对上述数据去重排序后, 重新写入到data文件夹下得2.txt文感谢件中, 即内容如下:

    12356911224.2 目得

    考察泛型, 列表, 流相关得内容.

    4.3 参考代码

    import java.io.{BufferedWriter, FileWriter}import scala.io.Source//案例: 列表去重排序, 并写入文件.object ClassDemo07 { def main(args: Array[String]): Unit = { //1. 定义数据源对象. val source = Source.fromFile("./data/1.txt") //2. 从指定文件中读取所有得数据(字符串形式) val list1:List[String] = source.mkString.split("\\s+").toList //3. 把List[String]列表转换成List[Int] val list2:List[Int] = list1.map(_.toInt) //4. 把List[Int]转换成Set[Int], 对列表元素去重. val set:Set[Int] = list2.toSet //5. 把Set[Int]转成List[Int], 然后升序排列 val list3:List[Int] = set.toList.sorted //println(list3) //6. 把数据重新写入到data文件夹下得2.txt文件中. val bw = new BufferedWriter(new FileWriter("./data/2.txt")) for(i <- list3) { bw.write(i.toString) bw.newline() //别忘记加换行 } //7. 释放资源 bw.close() }}

  •  
    (文/叶辰铭)
    打赏
    免责声明
    • 
    本文为叶辰铭原创作品•作者: 叶辰铭。欢迎转载,转载请注明原文出处:http://www.udxd.com/kbzx/show-119228.html 。本文仅代表作者个人观点,本站未对其内容进行核实,请读者仅做参考,如若文中涉及有违公德、触犯法律的内容,一经发现,立即删除,作者需自行承担相应责任。涉及到版权或其他问题,请及时联系我们邮件:weilaitui@qq.com。
     

    Copyright©2015-2023 粤公网安备 44030702000869号

    粤ICP备16078936号

    微信

    关注
    微信

    微信二维码

    WAP二维码

    客服

    联系
    客服

    联系客服:

    24在线QQ: 770665880

    客服电话: 020-82301567

    E_mail邮箱: weilaitui@qq.com

    微信公众号: weishitui

    韩瑞 小英 张泽

    工作时间:

    周一至周五: 08:00 - 24:00

    反馈

    用户
    反馈