从语法上来说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)
代码的执行结果如下:
在代码的第一行创建了一个名为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)
调用*()方法前,需要先调用了在左边的+()方法,但是因为*()方法优先,它会先执行,看一下执行结果:
可以看到*()方法先被执行了。
##########