- •Глава 2 3
- •Глава 3 11
- •Глава 10 117
- •Глава 2 Автоупаковка и автораспаковка
- •Обзор оболочек типов и упаковки значений
- •Основы автоупаковки/распаковки
- •Автоупаковка и методы
- •Автоупаковка/распаковка в выражениях
- •Автоупаковка/распаковка логических и символьных значений
- •Помощь автоупаковки/распаковки в предупреждении ошибок
- •Предостережения
- •Глава 3 Настраиваемые типы
- •Что такое настраиваемые типы
- •Простой пример применения настраиваемых типов
- •Средства настройки типов работают только с объектами
- •Различия настраиваемых типов, основанных на разных аргументах типа
- •Как настраиваемые типы улучшают типовую безопасность
- •Настраиваемый класс с двумя параметрами типа
- •Общий вид объявления настраиваемого класса
- •Ограниченные типы
- •Применение метасимвольных аргументов
- •Ограниченные метасимвольные аргументы
- •Создание настраиваемого метода
- •Настраиваемые конструкторы
- •Настраиваемые интерфейсы
- •Типы raw и разработанный ранее код
- •Иерархии настраиваемых классов
- •Использование настраиваемого суперкласса
- •Настраиваемый подкласс
- •Сравнения типов настраиваемой иерархии во время выполнения программы
- •Переопределенные методы в настраиваемом классе
- •Настраиваемые типы и коллекции
- •Стирание
- •Методы-подставки
- •Ошибки неоднозначности
- •Некоторые ограничения применения настраиваемых типов
- •Нельзя создавать объекты, используя параметры типа
- •Ограничения для статических членов класса
- •Ограничения для настраиваемого массива
- •Ограничение настраиваемых исключений
- •Заключительные замечания
- •Глава 4 Вариант For-Each цикла for
- •Описание цикла for-each
- •Обработка многомерных массивов в цикле
- •Область применения цикла for в стиле for-each
- •Использование цикла for в стиле for-each для обработки коллекций
- •Создание объектов, реализующих интерфейс Iterable
- •Глава 5 Аргументы переменной длины
- •Средство формирования списка с переменным числом аргументов
- •Перегрузка методов с аргументом переменной длины
- •Аргументы переменной длины и неоднозначность
- •Глава 6 Перечислимые типы
- •Описание перечислимого типа
- •Методы values() и valueOf()
- •Перечислимый тип в Java — это класс
- •Перечислимые типы, наследующие тип enum
- •Глава 7 Метаданные
- •Описание средства "метаданные"
- •Задание правил сохранения
- •Получение аннотаций во время выполнения программы с помощью рефлексии
- •Листинг 7.3. Получение всех аннотаций для класса и метода
- •Интерфейс AnnotatedElement
- •Использование значений по умолчанию
- •Аннотации-маркеры
- •Одночленные аннотации
- •Встроенные аннотации
- •Несколько ограничений
- •Глава 8 Статический импорт
- •Описание статического импорта
- •Общий вид оператора статического импорта
- •Импорт статических членов классов, созданных Вами
- •Неоднозначность
- •Предупреждение
- •Глава 9 Форматированный ввод/вывод
- •Форматирование вывода с помощью класса Formatter
- •Конструкторы класса Formatter
- •Методы класса Formatter
- •Основы форматирования
- •Форматирование строк и символов
- •Форматирование чисел
- •Форматирование времени и даты
- •Спецификаторы %n и %%
- •Задание минимальной ширины поля
- •Задание точности представления
- •Применение флагов форматирования
- •Выравнивание вывода
- •Флаг запятая
- •Применение верхнего регистра
- •Использование порядкового номера аргумента
- •Применение метода printf() языка Java
- •Класс Scanner
- •Конструкторы класса Scanner
- •Описание форматирования входных данных
- •Несколько примеров применения класса Scanner
- •Установка разделителей
- •Другие свойства класса Scanner
- •Глава 10 Изменения в api
- •Возможность применения настраиваемых типов при работе с коллекциями
- •Обновление класса Collections
- •Почему настраиваемые коллекции
- •Модернизация других классов и интерфейсов для применения настраиваемых типов
- •Новые классы и интерфейсы, добавленные в пакет java.Lang
- •Класс ProcessBulider
- •Класс StringBuilder
- •Интерфейс Appendable
- •Интерфейс Iterable
- •Интерфейс Readable
- •Новые методы побитной обработки классов Integer и Long
- •Методы signum() u reverseBytes()
- •Поддержка 32-битных кодовых точек для символов Unicode
- •Новые подпакеты пакета java.Lang
- •Классы Formatter и Scanner
Стирание
Обычно программисту не нужно знать, как компилятор языка Java превращает исходный текст программы в объектный код. Однако в случае настраиваемых типов некоторое общее представление об этом процессе важно, так как помогает понять, почему настраиваемые типы ведут себя так, а не иначе и почему иногда их поведение вызывает удивление. Именно поэтому уместно краткое обсуждение способа реализации настраиваемых типов в языке Java.
Необходимость сохранения совместимости с более ранними версиями Java существенно повлияла на способ введения в язык настраиваемых типов. Настраиваемый код должен быть совместим с уже существующим кодом, не использующим настраиваемые типы. Таким образом, любые изменения синтаксиса языка Java или виртуальной машины Java (JVM, Java Virtual Machine) не должны нарушать работоспособность уже действующего кода. Для удовлетворения этого ограничения в языке Java была выбрана реализация настраиваемых типов с помощью стирания (erasure).
В общем, стирание выполняется следующим образом. Когда компилируется исходный текст программы на языке Java, вся информация о настраиваемых типах удаляется (стирается). Это означает замену параметров типа ограничивающими их типами или верхними границами (bound type), например, типом Object при отсутствии явно заданной границы, и далее необходимое приведение типов (в соответствии с аргументами типа) для обеспечения совместимости с заданными аргументами типа. Компилятор также следит за совместимостью типов. Такой подход ведет к отсутствию какой-либо информации о настраиваемых типах во время выполнения приложения. Настраиваемые типы — только средство для разработки исходного текста программы. Для того чтобы лучше понять механизм стирания рассмотрим два класса, приведенные в листинге 3.19.
Листинг 3.19. Описание классов для демонстрации стирания
class Gen<T> {
T ob; // here, T will be replaced by Object
Gen(T o) {
ob = o;
}
// Return ob.
T getob() {
return ob;
}
}
// Here, T is bound by String.
class GenStr<T extends String> {
T str; // here, T will be replaced by String
GenStr(T o) {
str = o;
}
T getstr() { return str; }
}
Благодаря наличию стирания некоторые действия выполняются не так, как Вы могли бы себе представить. Например, рассмотрим короткую программу из листинга 3.20, в которой создаются два объекта настраиваемого класса Gen.
Листинг 3.20. Пример создания объектов настраиваемого класса
class GenTypeDemo {
public static void main(String args[]) {
Gen<Integer> iOb = new Gen<Integer>(99);
Gen<Float> fOb = new Gen<Float>(102.2F);
System.out.println(iOb.getClass().getName());
System.out.println(fOb.getClass().getName());
}
}
Далее приведен вывод программы:
Gen
Gen
Как видите, и у объекта iOb и у объекта fOb — тип Gen, а не Gen<Integer> и Gen<Float>, как Вы могли бы предположить. Запомните, все параметры типа стираются в процессе компиляции. Во время выполнения действительно существуют только несформированные (raw) типы.
