Предыдущий пример, T m 0, является примером "операторной нотации". Scala имеет три типа нотации операторов: префикс (называемый унарным), инфиксный и постфиксный. Давайте посмотрим примеры всех трех в действии здесь:
class MyByte(val n : Int) {
require(n >= 0 && n <= 255)
def unary_! = new MyByte(n ^ 0xff)
def +(m : MyByte) = new MyByte(n + m.n)
def bits = (math.log(n) / math.log(2) + 1).toInt
override def toString = "0" * (8 - bits) + n.toBinaryString
}
Здесь используется:
scala> val a = new MyByte(5)
a: MyByte = 00000101
scala> val b = new MyByte(10)
b: MyByte = 00001010
scala> ! a // Prefix/Unary
res58: MyByte = 11111010
scala> a + b // Infix
res59: MyByte = 00001111
scala> b bits // Postfix
res60: Int = 4
Несмотря на то, что инфиксная и постфиксная нотация принимает любой действительный идентификатор Scala, хотя и ведутся разговоры об ограничении постфиксной нотации, в качестве префикса могут использоваться только четыре идентификатора: ~,!, - и +.
Теперь, когда вы используете «m 0», Scala отбрасывает его как унарный оператор на том основании, что он недопустим (~,!, - и +). Он обнаруживает, что «m» является допустимым объектом - это функция, а не метод, и все функции являются объектами.
Поскольку «0» не является допустимым идентификатором Scala, он не может быть ни инфиксным, ни постфиксным оператором. Поэтому Скала жалуется, что ожидал ";" - которые будут разделять два (почти) допустимых выражения: «m» и «0». Если вы вставите его, то он будет жаловаться, что m требует либо аргумента, либо, в случае неудачи, «_», чтобы превратить его в частично примененную функцию.