scala中的隐式类型转换的实现

Scala语言中的隐式转换是一个十分强大的语言特性,主要可以起到两个作用:

成都创新互联公司长期为千余家客户提供的网站建设服务,团队从业经验10年,关注不同地域、不同群体,并针对不同对象提供差异化的产品和服务;打造开放共赢平台,与合作伙伴共同营造健康的互联网生态环境。为武乡企业提供专业的成都网站建设、网站制作武乡网站改版等技术服务。拥有十余年丰富建站经验和众多成功案例,为您定制开发。

一.自动进行某些数据类型的隐式转换

String类型是不能自动转换为Int类型的,所以当给一个Int类型的变量或常量赋予String类型的值时编译器将报错。所以,一下语句是错误的。

val x: Int = "100"

如果需要将一个字符串类型的整形数值赋给Int,比如使用String.toInt方法,例如:

val x: Int = "100".toInt

如果想让字符串自动转换为整形,就可以使用隐式转换。可以定义如下函数。

implicit def strToInt(str: String) = str.toInt

这时你再对Int类型的变量赋值字符串时,字符串就会自动转换为Int。

scala> val x:Int = "00"
x: Int = 100

如果你此时定义一个两数相加的函数

def add(x: Int, y: Int) = x + y

就可以达到这种效果:

scala> add("100", 200)
res1: Int = 300

隐式转换有一定的使用规则,比较重要的有2个。

1.按照《Scala编程》这本书中所说:插入的隐式转换必须以单一标识符的形式处于作用域中,或与转换的源或目标类型关联在一起。Scala编译器将仅考虑处于作用域之内的隐式转换。

简而言之,就是在使用隐式转换之前,需要用import把隐式转换引用到当前的作用域里或者就在作用域里定义隐式转换。除了隐式转换被引入进当前作用域之外,还有一种方式可以使用隐式转换,就是编译器会在源类型或者期望的伴生对象中寻找隐式定义。

2.无歧义规则:隐式转换只能在无其他可用转换的前提下才能操作。如果在同一作用域里,对同一源类型定义一个以上的隐式转换函数,如果多种隐式转换函数都可以匹配,那么编译器将报错,所以在使用时请移除不必要的隐式定义。

二.隐式参数

柯里化函数会有多个参数列表,当希望对某个参数列表采用默认参数时,可以使用implicit提供的隐式参数功能。做法是在需要自动填充的参数列表最开端加上implicit,然后在定义域内定义需要填充的默认参数值常量,并在常量的定义之前声明implicit。

视界

当有如下定义时

class Container[A <% Int] { def addIt(x: A) = 123 + x }

表示A类型必须可视为Int。简单的说,就是需要有一个转换函数,可以自动的将A类型,转换为Int类型,如果没有这样的转换函数,可以使用implicit定义。

写一个类测试一下Scala中的隐式转换的用法:

class Fraction(n: Int, d: Int) {
// def den = d
 private val den = d;
// def num = n 类参数定义为方法或字段都可以
 private val num = n;

// 定义乘法
 def *(other: Fraction) = Fraction(other.num * this.num, other.den * this.den)
// 重写toString
 override def toString() = s"$num / $den" 
}
//伴生对象
object Fraction {
// implicit隐转 方法名无关可以随意改,自动调用
 implicit def int2Fraction(n: Int) = Fraction(n, 1)

 def apply(n: Int, d: Int) = {
  new Fraction(n, d)
 }

 def unapply(frac: Fraction) = if (frac.den == 0) None else Some((frac.num, frac.den))
}

object TestFrac extends App{
// 3 隐式调用了int2Fraction方法被转化为一个Fraction对象Fraction(3,1)

 val result = 3 * Fraction(4,5)
// 也可以显示调用
 val result2 = Fraction.int2Fraction(5) * Fraction(3,4)
 println(result)
// unapply
 val Fraction(num,den) = result
 println(num,den)

}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持创新互联。


名称栏目:scala中的隐式类型转换的实现
转载来源:http://myzitong.com/article/jjihps.html