Второй этап обнаружения ошибок — когда компилятор встречает применение шаблона. На данном этапе компилятор также способен проверить немногое. Для вызова шаблона функции компилятор обычно проверяя количество и типы аргументов. Он может также проверить совпадение типов двух аргументов. Для шаблона класса компилятор может проверить количество и правильность предоставленных аргументов шаблона, но не более.
Третий этап обнаружения ошибок — момент создания экземпляра. Только теперь обнаруживаются ошибки, связанные с типами. В зависимости от того, как компилятор осуществляет создание экземпляра, он может сообщить об этих ошибках во время редактирования.
При написании шаблона код не может быть открыто специфическим для типа, но можно сделать некоторые предположения об используемых типах. Например, код первоначальной функции compare()
подразумевал, что тип аргумента имеет оператор <
.
if (v1 < v2) return -1; //
if (v2 < v1) return 1; //
return 0; //
Когда компилятор обрабатывает тело этого шаблона, он не может проверить корректность условий в операторах if
. Если переданные функции compare()
аргументы имеют оператор <
, то код сработает прекрасно, но не в противном случае. Например:
Sales_data data1, data2;
cout << compare(data1, data2) << endl; //
//
Этот вызов создает экземпляр функции compare()
с параметром Т
, замененным классом Sales_data
. Если условия попытаются использовать оператор <
для объектов класса Sales_data
, то окажется, что такого оператора нет. В результате получится экземпляр функции, которая не будет откомпилирована. Такие ошибки, как эта, не могут быть обнаружены, пока компилятор не создаст экземпляр определения функции compare()
для типа Sales_data
.
Упражнение 16.1. Определите создание экземпляра.
Упражнение 16.2. Напишите и проверьте собственные версии функций compare()
.
Упражнение 16.3. Вызовите собственную функцию compare()
для объекта класса Sales_data
и посмотрите, как ваш компилятор обрабатывает ошибки во время создания экземпляра.
Упражнение 16.4. Напишите шаблон, действующий как библиотечный алгоритм find()
. Функция будет нуждаться в двух параметрах типа шаблона: один — для представления параметров-итераторов функции и другой — для типа значения. Используйте свою функцию для поиска заданного значение в векторе vector
и списке list
.
Упражнение 16.5. Напишите шаблон функции print()
из раздела 6.2.4, которая получает ссылку на массив и может обрабатывать массивы любого размера и любого типа элементов.
Упражнение 16.6. Как работают библиотечные функции begin()
и end()
, получающие аргумент в виде массива? Определите собственные версии этих функций.
Упражнение 16.7. Напишите шаблон constexpr
, возвращающий размер заданного массива.
Упражнение 16.8. В разделе "Ключевая концепция" в разделе 3.4.1 упоминалось о том, что программисты С++ привыкли использовать оператор !=
, а не <
. Объясните причину этой привычки.
В качестве примера реализуем шаблонную версию класса StrBlob
(см. раздел 12.1.1). Присвоим шаблону имя Blob, указывающее, что он больше не специфичен только для строк. Как и класс StrBlob
, этот шаблон будет предоставлять совместный (и проверяемый) доступ к своим членам. В отличие от класса, шаблон применяется к элементам практически любого типа. Подобно библиотечным контейнерам, используя шаблон Blob
, пользователи должны будут определить тип элемента.