美文网首页
scala之隐式(implicit)

scala之隐式(implicit)

作者: 机器不能学习 | 来源:发表于2018-11-01 18:39 被阅读0次
隐式是scala的一大特性。

编写程序时使用它可以使代码精简。但是因为是隐式,所以在可读性上会受到一定影响。

隐式的分类

按照使用时机来分类,大概可以分为缺省参数,缺省对象,缺省类。
如果按照隐式的类别来分,可以分为隐式值,隐式视图,隐式类。

他们的对应关系大概如下
隐式值->缺省参数
隐式视图->缺省参数,缺省对象
隐式类->缺省对象

隐式值

scala> def max(implicit x:String) =x
max: (implicit x: String)String

scala> max
<console>:27: error: could not find implicit value for parameter x: String
       max
       ^

当我们定义一个方法,参数为隐式。我们去无参的调用它,这是编译器会去查找隐式值,但不幸,没有找到,报错。

scala> implicit var x="dsa"
x: String = dsa

scala> max
res35: String = dsa

现在我们定义了一个隐式值,再调用max,编译器可以找到一个对应的String隐式值,所以自动补上缺位。

scala> implicit var y="ha"
y: String = ha

scala> max
<console>:29: error: ambiguous implicit values:
 both method x of type => String
 and method y of type => String
 match expected type String
       max
       ^

这时我们也会想到,编译器是按类型进行查找,那么找到了多个String的隐式值怎么办?那么它会报错。

总结:当方法想以默认值执行的时候,可以应用隐式值。但是隐式值避免是String/int等基本类型,以免造成冲突。

隐式视图

首先我们定义一个分数,x是分子y是分母

class Rational(x:Int,y:Int) {
    require(y!=0)

    val up=x
    val down=y

      def add(r:Rational):Rational=
        new Rational(up*r.down+down*r.up,down*r.down)

}

现在如果我们现在调用该函数,但是我们给予的参数是一个整数

val oneHalf=new Rational(1,2)
    oneHalf+2

Error:(4, 13) type mismatch;
 found   : Int(2)
 required: Rational
    oneHalf+2

编辑器发现给的参数不对,会去寻找会不会有隐式转换,可惜没找到,现在我们给他一个隐式转换。

  def main(args: Array[String]): Unit = {
    implicit def intToRational(x:Int):Rational=new Rational(x,1)
    val oneHalf=new Rational(1,2)
    print(oneHalf+2)
  }

Rational@f6c48ac
Process finished with exit code 0

现在正确了,因为返回的是一个Rational,所以会是以@开头后面跟一串字符,可以重写方法中的toString方法,那么打印的会是toString的返回值。

现在我们希望有一个方法,可以返回一个分母加分子

class Rational(x:Int,y:Int) {
    require(y!=0)

    val up=x
    val down=y

      def +(r:Rational):Rational=
        new Rational(up*r.down+down*r.up,down*r.down)

      def addxy()=x+y
}

在一个类start中,它也希望能调用该方法

  val s=new start
    s.addxy()

Error:(6, 7) value addxy is not a member of start
    s.addxy()

不幸,没有找到。例子有些脑残。但是我们很容易想到这种情况可以用隐式解决。

implicit def sToRational(s:start):Rational=new Rational(1,1)      
 val s=new start                                                  
 s.addxy()                                                        

通过隐式转换,可以把s转为一个1/1的Rantional对象,从而进行调用。

联想:RDD中没有reduceByKey等函数,他们都在PairRDDFunctions中,所以使用函数时必须导入Import org.apache.spark.SparkContext._

这也不禁让我想到前些日子,调用JavaDStream的mapToPair对象无法返回JavaPairDStream
虽然导入了import org.apache.spark.streaming.api.java.JavaPairDStream;
但无法奏效,必须导入
import org.apache.spark.streaming.api.java.*;

总结:如果一个对象调用一个函数,参数不是该函数需要类型的参数,这是编译器会去寻找隐式转换。如果这个对象没有这个方法,那么编译器会去寻找一个有该方法的隐式对象。

隐式类

隐式类,可以在缺省类时被使用。还有动态增加函数的作用。在这里不细讲。

scala> implicit class StringSize(val s: String){
     | def sizes=s.length
     | }
defined class StringSize

scala> "/".sizes
res37: Int = 1

果然构造类参数换为其他对象的时候,就成了动态增加函数啦。

相关文章

网友评论

      本文标题:scala之隐式(implicit)

      本文链接:https://www.haomeiwen.com/subject/dvnxxqtx.html