scala中的隐式类型转换的实现
Scala语言中的隐式转换是一个十分强大的语言特性,主要可以起到两个作用:
一.自动进行某些数据类型的隐式转换
String类型是不能自动转换为Int类型的,所以当给一个Int类型的变量或常量赋予String类型的值时编译器将报错。所以,一下语句是错误的。
valx:Int="100"
如果需要将一个字符串类型的整形数值赋给Int,比如使用String.toInt方法,例如:
valx:Int="100".toInt
如果想让字符串自动转换为整形,就可以使用隐式转换。可以定义如下函数。
implicitdefstrToInt(str:String)=str.toInt
这时你再对Int类型的变量赋值字符串时,字符串就会自动转换为Int。
scala>valx:Int="00" x:Int=100
如果你此时定义一个两数相加的函数
defadd(x:Int,y:Int)=x+y
就可以达到这种效果:
scala>add("100",200) res1:Int=300
隐式转换有一定的使用规则,比较重要的有2个。
1.按照《Scala编程》这本书中所说:插入的隐式转换必须以单一标识符的形式处于作用域中,或与转换的源或目标类型关联在一起。Scala编译器将仅考虑处于作用域之内的隐式转换。
简而言之,就是在使用隐式转换之前,需要用import把隐式转换引用到当前的作用域里或者就在作用域里定义隐式转换。除了隐式转换被引入进当前作用域之外,还有一种方式可以使用隐式转换,就是编译器会在源类型或者期望的伴生对象中寻找隐式定义。
2.无歧义规则:隐式转换只能在无其他可用转换的前提下才能操作。如果在同一作用域里,对同一源类型定义一个以上的隐式转换函数,如果多种隐式转换函数都可以匹配,那么编译器将报错,所以在使用时请移除不必要的隐式定义。
二.隐式参数
柯里化函数会有多个参数列表,当希望对某个参数列表采用默认参数时,可以使用implicit提供的隐式参数功能。做法是在需要自动填充的参数列表最开端加上implicit,然后在定义域内定义需要填充的默认参数值常量,并在常量的定义之前声明implicit。
视界
当有如下定义时
classContainer[A<%Int]{defaddIt(x:A)=123+x}
表示A类型必须可视为Int。简单的说,就是需要有一个转换函数,可以自动的将A类型,转换为Int类型,如果没有这样的转换函数,可以使用implicit定义。
写一个类测试一下Scala中的隐式转换的用法:
classFraction(n:Int,d:Int){ //defden=d privatevalden=d; //defnum=n类参数定义为方法或字段都可以 privatevalnum=n; //定义乘法 def*(other:Fraction)=Fraction(other.num*this.num,other.den*this.den) //重写toString overridedeftoString()=s"$num/$den" } //伴生对象 objectFraction{ //implicit隐转方法名无关可以随意改,自动调用 implicitdefint2Fraction(n:Int)=Fraction(n,1) defapply(n:Int,d:Int)={ newFraction(n,d) } defunapply(frac:Fraction)=if(frac.den==0)NoneelseSome((frac.num,frac.den)) } objectTestFracextendsApp{ //3隐式调用了int2Fraction方法被转化为一个Fraction对象Fraction(3,1) valresult=3*Fraction(4,5) //也可以显示调用 valresult2=Fraction.int2Fraction(5)*Fraction(3,4) println(result) //unapply valFraction(num,den)=result println(num,den) }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。