Deletions are marked like this. | Additions are marked like this. |
Line 39: | Line 39: |
{{{ class Zoo<T extends Animal> { public void add(T animal) { } } .... Zoo<Fish>fishZoo = new Zoo<Fish>(); // OK Zoo<Integer>intZoo = new Zoo<Integer>(); // エラー }}} Scalaでは {{{ class Zoo[T <: Animal] { def add(a:T) { } } .... val fishZoo = new Zoo[Fish]; // OK val intZoo = new Zoo[Int]; // エラー }}} |
Scalaの型システム
単純な型パラメータ
Javaでは以下のように書ける。
class Animal { } class Fish extends Animal { } class Zoo<T> { public void add(T animal) { } } ... Zoo<Animal>zoo = new Zoo<Animal>(); zoo.add(new Fish()); ...
これと等価な(たぶん)Scalaコードは以下。
class Animal class Fish extends Animal; class Zoo[T] { def add(a:T) { } } val zoo = new Zoo[Animal]; zoo add new Fish
上限境界と下限境界
上のコードのままだと、Animal以外のZooの作成が可能になってしまうし、そもそもZooクラスの中でAnimalのメソッドを呼び出せない。なぜなら、Tという型に制限がないので、IntegerやDoubleという型パラメータを指定してZooを作成できてしまうからである。
Zoo<Integer>zooInt = new Zoo<Integer>(); // 可能だが???
そこで、TというクラスをAnimalあるいはそのサブクラスに制限する。これを上限境界という。
Javaでは
class Zoo<T extends Animal> { public void add(T animal) { } } .... Zoo<Fish>fishZoo = new Zoo<Fish>(); // OK Zoo<Integer>intZoo = new Zoo<Integer>(); // エラー
Scalaでは
class Zoo[T <: Animal] { def add(a:T) { } } .... val fishZoo = new Zoo[Fish]; // OK val intZoo = new Zoo[Int]; // エラー