scala学习手记7 - 运算符重载

从语法上来说scala是没有运算符的。之前的一节里也曾提到过scala的运算符实际上是方法名,如1 + 2实际上就是1.+(2)。我们可以将之视为运算符,是因为scala的一个特性:如果方法的参数小于等于1个的话,那么“.”和括号就都是可选的。

scala的运算符重载指的就是重载+、-这样的符号——和C、java或者python等语言不一样,我们需要自己定义这些符号如何实现。

下面看一个“+”运算符实现的例子。这里定义了一个Complex类,Complex指的是复数。复数有实部和虚部,计算时需要分别处理:

class Complex(val real: Int, val imaginary: Int) {
  def +(operand: Complex): Complex = {
    new Complex(real + operand.real, imaginary + operand.imaginary)
  }

  override def toString(): String = {
    real + (if (imaginary < 0) "" else "+") + imaginary + "i"
  }
}

val c1 = new Complex(1, 2)
val c2 = new Complex(2, -3)
val sum = c1 + c2
println("(" + c1+ ")+ (" + c2 + ")=" + sum)

代码的执行结果如下:

image

在代码的第一行创建了一个名为Complex的类、定义了一个有两个参数的构造器。

在+方法里创建了一个新的Complex实例作为计算结果。结果的实部和虚部分别对应两个运算数实部和虚部之和。计算时c1+c2实际上就是c1.+(c2)。

既然scala中看起来像运算符的东西都是方法,那么和运算符相关的一些内容,比如算数优先级,scala是怎样处理的呢?scala中确实没有算数优先级、但是它定义了方法优先级:方法的优先级由方法名的第一个字符决定,如果表达式里有两个相同优先级的方法,那么左边的方法优先级更高。(scala的这个设定蛮有趣的,解决了算数优先级的问题,而且说不定什么地方就能带出一些惊喜来)。下面从低到高列出了scala中方法名首字符的优先级

|
^
&
< >
=  !
: 
+   - 
*  /  % 
所有其他特殊字符

下面看一个示例程序。在这个示例程序里为之前的Complex类添加了乘法运算方法:

class Complex(val real: Int, val imaginary: Int) {

  def +(operand: Complex): Complex = {
    println("Calling +")
    new Complex(real + operand.real, imaginary + operand.imaginary)
  }


  def *(operand: Complex): Complex = {
    println("Calling *")
    new Complex(real * operand.real - imaginary * operand.imaginary, real * operand.imaginary + imaginary * operand.real)
  }

  override def toString(): String = {
    real + (if (imaginary < 0) "" else "+") + imaginary + "i"
  }
}

val c1 = new Complex(1, 4)
val c2 = new Complex(2, -3)
val c3 = new Complex(2, 2)
println(c1 + c2 * c3)

调用*()方法前,需要先调用了在左边的+()方法,但是因为*()方法优先,它会先执行,看一下执行结果:

image

可以看到*()方法先被执行了。

##########

posted @ 2016-06-07 22:58  robin·张  阅读(1795)  评论(0编辑  收藏  举报