Scala隐式转换与隐式参数

我们有时需要类A具有类B的方法,可以自定义一个隐式转换,将类A转换为类B,在引入这个隐式转换的作用域,使用类A就相当于使用了类B,
这样就达到了增强类A功能的目的。类似于装饰模式。

class RichFile(val file:File){
   def read = Source.fromFile(file.getPath()).mkString
}
 
object Context{
    implicit def file2RichFile(file:File)= new RichFile(file) //File -> RichFile,必须有implicit关键字
}

object Hello_Implicit_Conversions {

  def main(args: Array[String]) {
        import Context.file2RichFile //在当前作用域引入隐式转换
     println(new File("D:\\hello.txt").read)  //File本身是没有read方法的,需要隐式转换为自定义的RichFile
  }
}

隐式参数,最直观的是,参数前面有implicit关键字,编译器会在作用域范围内查找谁是implict类型的值,如果有,则作为缺省值。

object Context_Implicits{
    implicit val default:String = "Flink" //存在一个隐式值
}
 
object Param{
   def print(content:String)(implicit language:String){
      println(language+":"+content)
   }
}
object Implicit_Parameters {

  def main(args: Array[String]) {
    Param.print("Spark")("Scala")
     
    import Context_Implicits._ //从作用域范围内或者Object伴生对象中查找隐式参数
    Param.print("Hadoop") //使用隐式参数
  }
}

隐式参数下的隐式转换

object Implicit_Conversions_with_Implicit_Parameters {
  def main(args: Array[String]) {
    def bigger[T](a: T, b: T)(implicit ordered: T => Ordered[T]) //将匿名函数赋值给ordered变量,该隐式参数将T类型转换为Ordered[T]类型
        = if (a > b) a else b //Ordered实现了<, >等
    println(bigger(4,3))
    println(bigger(4.2,3))
    println(bigger("Spark","Hadoop"))
  }
}

上下文界定中的隐式参数

class Pair_Implicits[T: Ordering](val first: T, val second: T){ //Ordering是泛型,上下文界定说明,只要存在T:Ordering ,就会存在一个隐式值Ordering[T]
  def bigger(implicit ordered: Ordering[T]) =
    if (ordered.compare(first, second) > 0) first else second
}
class Pair_Implicitly[T: Ordering](val first: T, val second: T){
    def bigger =if (implicitly[Ordering[T]].compare(first, second) > 0)  //implicitly提取出运行时上下文界定的实例
            first else second
}
class Pair_Implicitly_Odereded[T: Ordering](val first: T, val second: T){
    def bigger ={
      import Ordered._
      if (first > second) first else second //将Ordering[T]转换为Ordered[T],可以直接使用>
    }    
}

object Context_Bounds_Internals {
  def main(args: Array[String]){
    println(new Pair_Implicits(7, 9).bigger)
    println(new Pair_Implicitly(7, 9).bigger)
    println(new Pair_Implicitly_Odereded(7, 9).bigger)
  }
}

隐式类:对类的功能增强

object Context_Helper{ 
    implicit class FileEnhancer(file : File){ 
        def read = Source.fromFile(file.getPath).mkString 
    }
    implicit class Op(x:Int){
         def addSAP(second: Int) = x + second
    }
}
object Implicits_Class {
  def main(args: Array[String]){
    import Context_Helper._
    println(1.addSAP(2))
    println(new File("E:\\Test.txt").read) 
    
  } 
}

隐式对象

abstract class Template[T] {
      def add(x: T, y: T): T
    }
abstract class SubTemplate[T] extends Template[T] {
      def unit: T
    }
object Implicits_Object {

  def main(args: Array[String]) {
     implicit object StringAdd extends SubTemplate[String] {
        override def add(x: String, y: String) = x concat y
        override def unit: String = ""
      }
      implicit object IntAdd extends SubTemplate[Int] {
        override def add(x: Int, y: Int) = x + y
        override def unit: Int = 0
      }
      def sum[T](xs: List[T])(implicit m: SubTemplate[T]): T = //m为隐式对象,根据T的具体类型引入相应的隐式对象
      
      if (xs.isEmpty) m.unit
      else m.add(xs.head, sum(xs.tail))  

      println(sum(List(1, 2, 3, 4, 5))) //引入IntAdd隐式对象
      println(sum(List("Scala", "Spark", "Kafka"))) //引入StringAdd隐式对象
    
  }
}

隐式转换内幕

class RicherFile(val file:File){
   def read = Source.fromFile(file.getPath()).mkString
}
 
class File_Implicits( path: String) extends File(path)
object File_Implicits{
    implicit def file2RicherFile(file:File)= new RicherFile(file) //File -> RicherFile
}

object Implicits_Internals {
    def main(args: Array[String]) {

        //因为类可以访问伴生对象里的所有内容,所以这里不需要import
        println(new File_Implicits("E:\\Test.txt").read) //当前作用域没有import隐式参数,尝试从类的伴生对象中查找 
    }
}

你可能感兴趣的:(Scala隐式转换与隐式参数)