Студопедия.Орг Главная | Случайная страница | Контакты | Мы поможем в написании вашей работы!  
 

Шаблоны функций



Рассмотрим простую функцию, реализующую алгоритм сравнения двух величин:

int minimum (int iVal_1, int iVal_2){ return iVal_1 < iVal_2? iVal_1: iVal_2; /* Возвращается значение iVal_1, если это значение меньше iVal_2. В противном случае возвращается значение iVal_2. */}

Для каждого типа сравниваемых величин должен быть определен собственный вариант функции min(). Вот как эта функция выглядит для float:

float minimum (float fVal_1, float fVal_2){ return fVal_1 < fVal_2? fVal_1: fVal_2;}

А для double… А для…

И так для всех используемых в программе типов сравниваемых величин. Мы можем бесконечно упражняться в создании совместно используемых функций, хотя можно воспользоваться средствами препроцессирования:

#define minimum (a,b) ((a)<(b)?(a):(b))

Это определение правильно работает в простых случаях:

minimum (10, 20);minimum (10.0, 20.0);

В более сложных случаях могут получаться неожиданные результаты. Это происходит из-за того, что препроцессор действует независимо от компилятора, до компилятора и вообще производит лишь простую текстовую обработку исходного модуля.

C++ предоставляет еще одно средство для решения этой задачи. При этом сохраняется присущая макроопределениям краткость и строгость контроля типов языка. Этим средством является шаблон функции.

Шаблон функции позволяет определять семейство функций. Это семейство характеризуется общим алгоритмом, который может применяться к данным различных типов. При этом задание конкретного типа данных для очередного варианта функции обеспечивается специальной синтаксической конструкцией, называемой списком параметров шаблона функции. Объявление функции, которому предшествует список параметров шаблона, называется шаблоном функции.

Объявление и определение шаблона функции начинается ключевым словом template, за которым следует заключенный в угловые скобки и разделенный запятыми непустой список параметров шаблона. Эта часть объявления или определения обычно называется заголовком шаблона.

Каждый параметр шаблона состоит из служебного слова class, за которым следует идентификатор. В контексте объявления шаблона функции служебное слово class не несет никакой особой смысловой нагрузки. Дело в том, что аналогичная конструкция используется также и для объявления шаблона класса, где, ключевое слово class играет свою особую роль. В заголовке шаблона имена параметров шаблона должны быть уникальны.

Следом за заголовком шаблона располагается прототип или определение функции — все зависит от контекста программы. Как известно, у прототипа и определения функции также имеется собственный заголовок. Этот заголовок состоит из спецификатора возвращаемого значения (вполне возможно, что спецификатором возвращаемого значения может оказаться идентификатор из списка параметров шаблона), имя функции и список параметров. Все до одного идентификаторы из заголовка шаблона обязаны входить в список параметров функции. В этом списке они играют роль спецификаторов типа. Объявления параметров, у которых в качестве спецификатора типа используется идентификатор из списка параметров шаблона, называется шаблонным параметром. Наряду с шаблонными параметрами в список параметров функции могут также входить параметры основных и производных типов.

Шаблон функции служит инструкцией для транслятора. По этой инструкции транслятор может самостоятельно построить определение новой функции.

Параметры шаблона в шаблонных параметрах функции обозначают места будущей подстановки, которую осуществляет транслятор в процессе построения функции. Область действия параметров шаблона ограничивается шаблоном. Поэтому в различных шаблонах разрешено использование одних и тех же идентификаторов-имен параметров шаблона.

В качестве примера рассмотрим программу, в которой для определения минимального значения используется шаблон функции min().

template <class Type> Type minimum (Type a, Type b);/*Прототип шаблона функции.Ключевое слово template обозначает начало списка параметров шаблона. Этот список содержит единственный идентификатор Type. Сама функция содержит два объявления шаблонных параметра, специфицированных шаблоном параметра Type.Спецификация возвращаемого значения также представлена шаблоном параметра Type.*/int main (void){ minimum (10,20);// int minimum (int, int); minimum (10.0,20.0);// float minimum (float, float); /* Вызовы шаблонной функции. Тип значений параметров определен. На основе выражения вызова (транслятор должен распознать тип параметров) и определения шаблона транслятор самостоятельно строит различные определения шаблонных функций. И только после этого обеспечивает передачу управления новорожденной шаблонной функции. */ return 1;}template <class Type>Type minimum (Type a, Type b){ return a < b? a: b;}/*По аналогии с определением функции, эту конструкцию будем называтьопределением шаблона функции.*/

Определение шаблона функции заставляет транслятор самостоятельно достраивать определения новых шаблонных функций, а точнее, создавать множество совместно используемых функций, у которых типы параметров и, возможно, тип возвращаемого значения зависит от типа параметров и типа возвращаемого значения в вызовах шаблонной функции. Этот процесс определения называют конкретизацией шаблона функции.

В результате конкретизации шаблона функции min() транслятор строится следующий вариант программы с двумя шаблонными функциями. По выражению вызова на основе шаблона строится шаблонная функция. Следует обратить внимание на то, что шаблон функции и шаблонная функция — два разных понятия.

int minimum (int a, int b);float minimum (float a, float b);int main (void){ minimum(10, 20); minimum(10.0, 20.0); return 1;}int minimum(int a, int b){ return a < b? a: b;}float minimum(float a, float b){ return a < b? a: b;}

Построение шаблонной функции осуществляется на основе выражений вызова. При этом в качестве значений параметров в выражении вызова могут быть использованы значения любых типов, для которых определены используемые в теле функции операции. Так, для функции minimum() тип параметров зависит от области определения операции сравнения <.

Типы формального параметра шаблона и значения параметра выражения вызова сопоставляются без учета каких-либо модификаторов типа. Например, если параметр шаблона в определении функции объявлен как

template <class Type>Type minimum(Type *a, Type *b){ return *a < *b? *a: *b;}и при этом вызов функции имеет вид:int a = 10, b = 20;int *pa = &a, *pb = &b;minimum(pa, pb);

то в процессе конкретизации идентификатор типа Type будет замещен именем производного типа int:

int minimum(int *a, int *b){ return a < b? a: b;}

В процессе конкретизации недопустимы расширения типов и другие преобразования типов параметров:

template <class Type>Type minimum(Type a, Type b){ return a < b? a: b;}unsigned int a = 10;:::::minimum(1024,a);/*Здесь транслятор сообщит об ошибке. В вызове функции тип второго фактического параметра модифицирован по сравнению с типом первого параметра - int и unsigned int. Это недопустимо. В процессе построения новой функции транслятор не распознает модификации типов. В вызове функции типы параметров должны совпадать. Исправить ошибку можно с помощью явного приведения первого параметра.*/min((unsigned int)1024,a);:::::

Имя параметра шаблона в определяемой функции используется в качестве имени типа. С его помощью специализируются формальные параметры, определяется тип возвращаемого значения, определяется тип объектов, локализованных в теле функции. Имя параметра шаблона скрывает объекты с аналогичным именем в глобальной по отношению к определению шаблонной функции области видимости. Если в теле шаблонной функции необходим доступ к внешним объектам с тем же именем, следует использовать операцию изменения области видимости.

Рассмотрим пример с классом ComplexType. На множестве комплексных чисел определены лишь два отношения: равенства (предполагает одновременное равенство действительной и мнимой частей) и неравенства (предполагает все остальные случаи). В нашей новой программе мы объявим и определим шаблон функции neq(), которая будет проверять на неравенство значения различных типов.

Для того, чтобы построить шаблонную функцию neq() для комплексных чисел, нам придется дополнительно определить операторную функцию-имитатор операции!= для объектов-представителей множества комплексных чисел. Это важно, поскольку операция!= явным образом задействована в шаблоне neq(). Транслятор не поймет, как трактовать символ!=, а, значит, и как строить шаблонную функцию neq(ComplexType, ComplexType), если эта операторная функция не будет определена для класса ComplexType.

#include <iostream.h>template <class Type>int neq (Type, Type); /*Прототип шаблона функции.*/class ComplexType{public: double real; double imag;// Конструктор умолчания. ComplexType(double re = 0.0, double im = 0.0) {real = re; imag = im;}/*Операторная функция!=. Без нее невозможно построение шаблонной функции neq() для комплексных чисел.*/ int operator!= (ComplexType &KeyVal) { return!(real == KeyVal.real && imag == KeyVal.imag); }};void main (){ // Определены и проинициализированы переменные трех типов. int i = 1, j = 2; float k = 1.0, l = 2.0; ComplexType CTw1(1.0,1.0), CTw2(2.0,2.0); //На основе выражений вызова транслятор строит три шаблонных функции. cout << "neq() for int:" << neq(i,j) << endl; cout << "neq() for float:" << neq(k,l) << endl; cout << "neq() for ComplexType:" << neq(CTw1, CTw2) << endl;}/*Определение шаблона функции.*/template <class Type>int neq (Type a, Type b){ return a!= b; }

И еще один пример. Этот пример подтверждает обязательность включения всех параметров шаблона в список параметров шаблона определяемой функции. Независимо от того, какая роль предназначается шаблонному параметру (он вообще может не использоваться в шаблонной функции), его присутствие в списке параметров обязательно. В процессе построения шаблонной функции транслятор модифицирует весь шаблон полностью — его заголовок и его тело. Так что в теле шаблона можно объявлять переменные, специфицированные параметрами шаблона.

#include <iostream.h>#include <typeinfo.h> /*В программе используется объект класса Type_info, позволяющий получать информацию о типе. Здесь подключается заголовочный файл, содержащий объявление этого класса */template <class YYY, class ZZZ> YYY Tf (ZZZ, YYY, int);/*Шаблон прототипа функции. Функция Tf возвращает значение пока еще неопределенного типа, обозначенного параметром шаблона YYY. Список ее параметров представлен двумя (всеми!) параметрами шаблона и одним параметром типа int.*/void main(){ cout << Tf((int) 0, '1', 10) << endl;/*Собственно эти вызовы и управляют работой транслятора. Тип передаваемых значений параметров предопределяет структуру шаблонной функции. В первом случае шаблону параметра ZZZ присваивается значение "int", шаблону параметра YYY присваивается значение "char", после чего прототип шаблонной функции принимает вид char Tf (int, char, int);*/ cout << Tf((float) 0, "This is the string...", 10) << endl;/*Во втором случае шаблону параметра ZZZ присваивается значение "float", шаблону параметра YYY присваивается значение "char *", после чего прототип шаблонной функции принимает вид char* Tf (float, char *, int); В результате, используя один общий шаблон, мы получаем две совершенно различных совместно используемых функции.*/}/*Шаблон функции. Первый параметр не используется, поэтому в списке параметров он представлен спецификатором объявления. Второй шаблонный параметр определен и также зависит от шаблона, третий параметр от шаблона не зависит.*/template <class YYY, class ZZZ> YYY Tf (ZZZ, YYY yyyVal, int x){ ZZZ zzzVal; int i; for (i = 0; i < x; i++) { cout << "Tf() for " << typeid(zzzVal).name() << endl; } return yyyVal;}

Вывод для приведенного примера:





Дата публикования: 2015-01-13; Прочитано: 268 | Нарушение авторского права страницы | Мы поможем в написании вашей работы!



studopedia.org - Студопедия.Орг - 2014-2024 год. Студопедия не является автором материалов, которые размещены. Но предоставляет возможность бесплатного использования (0.006 с)...