Scala 的類大抵和 Java 是類似的,簡單的例子如下:
class MyClass { var myField : Int = 0; def this(value : Int) = { this(); this.myField = value; } def getMyField() : Int = { return this.myField; } def addToMyField(value : Int) { this.myField += value; } }
如果好奇,可以把編譯後的程式碼反編譯成Java程式碼看看:
import scala.reflect.ScalaSignature; @ScalaSignature(bytes="\006\00112A!\001\002\001\013\t9Q*_\"mCN\034(\"A\002\002\017q*W\016\035;z}\r\0011C\001\001\007!\t9!\"D\001\t\025\005I\021!B:dC2\f\027BA\006\t\005\031\te.\037*fM\")Q\002\001C\001\035\0051A(\0338jiz\"\022a\004\t\003!\001i\021A\001\005\b%\001\001\r\021\"\001\024\003\035i\027PR5fY\022,\022\001\006\t\003\017UI!A\006\005\003\007%sG\017C\004\031\001\001\007I\021A\r\002\0275Lh)[3mI~#S-\035\013\0035u\001\"aB\016\n\005qA!\001B+oSRDqAH\f\002\002\003\007A#A\002yIEBa\001\t\001!B\023!\022\001C7z\r&,G\016\032\021\t\0135\001A\021\001\022\025\005=\031\003\"\002\023\"\001\004!\022!\002<bYV,\007\"\002\024\001\t\0039\023AC4fi6Kh)[3mIR\tA\003C\003*\001\021\005!&\001\007bI\022$v.T=GS\026dG\r\006\002\033W!)A\005\013a\001)\001") public class MyClass { private int myField = 0; public int myField() { return this.myField; } public void myField_$eq(int x$1) { this.myField = x$1; } public int getMyField() { return myField(); } public void addToMyField(int value) { myField_$eq(myField() + value); } public MyClass() { } public MyClass(int value) { this(); myField_$eq(value); } }
除了ScalaSignature,其他部分再熟悉不過了。只是Scala在語法上稍作了改變,在學習了golang和swift之後,也應該習以為常了吧。
類部分雖然沒有太多可說的地方,但是在物件上就有些許不同。Scala 中 object 用於構建 singleton 型別。
object Main { def sayHi() { println("Hi!"); } }
在 Java 進行呼叫的話,就是
Main.sayHi();
之前在Java中寫過Singleton程式碼的話,很容易猜出來具體的實現。反編譯的程式碼:
import scala.reflect.ScalaSignature; @ScalaSignature(bytes="\006\001e9Q!\001\002\t\002\025\tA!T1j]*\t1!A\004=K6\004H/\037 \004\001A\021aaB\007\002\005\031)\001B\001E\001\023\t!Q*Y5o'\t9!\002\005\002\f\0355\tABC\001\016\003\025\0318-\0317b\023\tyAB\001\004B]f\024VM\032\005\006#\035!\tAE\001\007y%t\027\016\036 \025\003\025AQ\001F\004\005\002U\tQa]1z\021&$\022A\006\t\003\027]I!\001\007\007\003\tUs\027\016\036") public final class Main { public static void sayHi() { Main..MODULE$.sayHi(); } }
實現上來講,只是多了一個 Main$ 類來實現具體的程式碼。
有了靜態成員,如何新增非靜態成員?Scala給出的答案是 Companion Objects,也就是定義同名 class。
object Main { def sayHi() { println("Hi!"); } } class Main { def sayHelloWorld() { println("Hello World"); } }
這樣一來,Main既支援靜態方式呼叫,也支援例項化。其實在 Java 中只是一個類,包含了靜態成員和非靜態成員而已。
public class Main { public static void sayHi() { Main..MODULE$.sayHi(); } public void sayHelloWorld() { Predef..MODULE$.println("Hello World"); } }
可以考慮一種情況,object 和 class 中都定義了相同名稱的成員,結果會如何呢?
object Main { def sayHi() { println("Hi!"); } } class Main { def sayHelloWorld() { println("Hello World"); } def sayHi() { println("non-static hi!") } }
編譯沒有報錯,結果是靜態成員無法訪問了。
error: non-static method sayHi() cannot be referenced from a static context Main.sayHi(); ^ 1 error
靜態方法被隱藏了,可以通過反編譯來驗證。
public class Main { public void sayHelloWorld() { Predef..MODULE$.println("Hello World"); } public void sayHi() { Predef..MODULE$.println("non-static hi!"); } }
實際生成的程式碼中並不包含靜態成員。有沒有辦法訪問之前的靜態成員呢?可以試試下面的程式碼:
Main$.MODULE$.sayHi();
當然,以上是Java中呼叫的情況,在 Scala 中是可以區分兩個sayHi的。
object UseInScala { def main(args: Array[String]) { Main.sayHi(); var main = new Main(); main.sayHi(); } }
輸出:
> scala UseInScala Hi! non-static hi!
從這些方面來看,Scala 的程式碼是極容易引入到 Java 中的,下一步可能會花些功夫在 Scala 的學習和應用上。
[1]http://tutorials.jenkov.com/scala/singleton-and-companion-objects.html