Из этих примеров видно, что неопределенные типы S, T могут быть типами параметров конструкторов и типами возвращаемых методами значений. Разумеется, они могут быть типами параметров не только конструкторов, но и любых методов. Более того, типами параметров и типами возвращаемых значений методов могут быть настраиваемые типы. Можно написать метод в такой форме:
public MyGenericClass2 makeClass2(S id,
MyGenericClass
}
и обратиться к нему так, как показано в листинге 4.3.
public class MyGenericClass2Demo{
public MyGenericClass2
makeClass2(S id, MyGenericClass
return new MyGenericClass2(id, data.getData());
}
public static void main(String[] args){
MyGenericClass
MyGenericClass2Demo
MyGenericClass2
}
}
В предыдущих главах мы часто пользовались тем, что можно определить ссылку типа суперкласса, ссылающуюся на объект подкласса, например:
Number n = new Long(123456L);
Number d = new Double(27.346);
Более того, это свойство распространяется на массивы:
Number[] n = new Long[100];
Можно ли распространить эту возможность на настраиваемые типы? Например, можно ли написать последний оператор листинга 4.3 так:
MyGenericClass2
d.makeClass2(123456L, dMyGen);
Ответ отрицательный. Из того, что какой-то класс B является подклассом класса A, не следует, что класс g будет подклассом класса g.
Это непривычное обстоятельство вынудило ввести дополнительную конструкцию —
MyGenericClass2 extends Number, ? extends Number> n = // Верно.
d.makeClass2(123456L, dMyGen);
или
MyGenericClass2
d.makeClass2(123456L, dMyGen);
Можно написать даже неограниченный шаблон типа
MyGenericClass2, ?> n =
d.makeClass2(123456L, dMyGen);
Такая запись будет почти эквивалентна записи
MyGenericClass2 n =
d.makeClass2(123456L, dMyGen);
за тем исключением, что в первом случае компилятор сделает более строгие проверки.
Кроме записи extends Type>, означающей "произвольный подтип типа Type, включая сам тип Type", можно написать выражение super Type>, означающее "произвольный супертип типа Type, включая сам тип Type".
Шаблон типа можно использовать в тех местах кода, где настраивается тип, в том числе в параметрах метода:
public MyGenericClass2 makeClass2(S id,
MyGenericClass extends Number> data){
return new MyGenericClass2(id, data.getData());
}
но, поскольку шаблон типа не является типом, его нельзя применять для создания объектов и массивов. Следующие определения неверны:
Average extends Number> a = // Ошибка!
new Average extends Number>(iArray);
Average extends Number>[] a = // Ошибка!
new Average extends Number>[10];
Тем не менее при определении массива (но не объекта) можно записать неограниченный шаблон типа:
Average extends Number>[] a = // Верно.
new Average >[10];
Настраиваемые методы
Настраиваемыми могут быть не только типы, но и методы. Параметры настраиваемого метода (type parameters) указываются в заголовке метода в угловых скобках перед типом возвращаемого значения. Это выглядит так, как показано в листинге 4.4.
public class MyGenericClass2Demo{
public MyGenericClass2
makeClass2(S id, MyGenericClass
return new MyGenericClass2(id, data.getData());
} public static void main(String[] args){
MyGenericClass
MyGenericClass2Demo d =
new MyGenericClass2Demo();
MyGenericClass2
}
}
Метод makeClass2 () описан в простом, ненастраиваемом, классе MyGenericClass2Demo, и его параметры задаются в угловых скобках . Здесь можно записывать ограниченные параметры
public
MyGenericClass2 makeClass2(S id, MyGenericClass
return new MyGenericClass2(id, data.getData());
}
Как видно из листинга 4.4, специально настраивать метод не нужно, конкретные типы его параметров и возвращаемого значения определяются компилятором по переданным в метод аргументам.