68956

Шаблони. Функція з двома узагальненими параметрами

Лекция

Информатика, кибернетика и программирование

За допомогою шаблонів можна створювати узагальнені функції і класи які працюють з типом даних заданим як параметр. Узагальнені функції Узагальнена функція визначає універсальну сукупність операцій застосовних до різних типів даних. За допомогою узагальненої функції можна визначити природу...

Украинкский

2014-09-28

54.5 KB

0 чел.

Лекція № 18

Тема: Шаблони

План

  1.  Узагальнені функції
  2.  Функція з двома узагальненими параметрами
  3.  Явне перевантаження узагальненої функції

Шаблони — один з найбільш складних і могутніх механізмів мови C++. Їх не було в первинній версії мови C++, але через декілька років вони стали його невід'ємною частиною, підтримуваною всіма сучасними компіляторами. За допомогою шаблонів можна створювати узагальнені функції і класи, які працюють з типом даних, заданим як параметр. Таким чином, одну і ту ж функцію або клас можна застосовувати до різних типів даним, не використовуючи окремі варіанти для кожного типу.

Узагальнені функції

Узагальнена функція визначає універсальну сукупність операцій, застосовних до різних типів даних. Тип даних, з якими працює функція, передається як параметр. Це дозволяє застосовувати одну і ту ж функцію до широкого спектру даних. Як відомо, багато алгоритмів носять універсальний характер і не залежать від типу даних, якими вони оперують. Наприклад, для масивів цілих і дійсних чисел використовується один алгоритм швидкого сортування. За допомогою узагальненої функції можна визначити природу алгоритму незалежно від типу даних. Після цього компілятор автоматично генерує правильний код, відповідний конкретному типу. По суті, узагальнена функція автоматично перенавантажує саму себе.

Узагальнена функція оголошується за допомогою ключового слова template. Звичайне значення слова "template" (шаблон) точно відповідає своєму сенсу в мові C++. Воно використовується для створення шаблону, який описує дії функції і дозволяє компілятору самому уточнювати необхідні деталі. Визначення шаблонної функції виглядає таким чином.

template <class Tтип> тип_возвращаемого_значения

имя_функции (список_параметров)

{

/ / Тіло функції

}

Тут параметр Ттіп задає тип даних, з яким працює функція. Цей параметр можна використовувати і усередині функції, проте при створенні конкретної версії узагальненої функції компілятор автоматично підставить замість нього фактичний тип. Традиційно узагальнений тип задається за допомогою ключового слова class, хоча замість нього можна застосовувати ключове слово typename.

У наступному прикладі демонструється узагальнена функція, що міняє місцями дві змінні. Оскільки процес перестановки не залежить від типу змінних, його можна описати за допомогою узагальненої функції.

// Приклад шаблонної функції.

#include <iostream>

using namespace std;

// шаблон функції

Template <class X> void swapargs(X &a, X &b)

{

X temp;

temp=a; a=b; b=temp;

}

int main()

{

int i=10, j=20;

double x=10.1, y=23.3;

char a=’x’, b=’z’;

cout << “Вихідні значення i, j: ” << i << ‘ ’ << j << ‘\n’;

cout << “Вихідні значення x, y: ” << x << ‘ ’ << y << ‘\n’;

cout << “Вихідні значення a, b: ” << a << ‘ ’ << b << ‘\n’;

swapargs(i, j);

swapargs(x,y);

swapargs(a,b);

cout << “Змінені значення i, j: ” << i << ‘ ’ << j << ‘\n’;

cout << “Змінені значення x, y: ” << x << ‘ ’ << y << ‘\n’;

cout << “Змінені значення a, b: ” << a << ‘ ’ << b << ‘\n’;

return 0;

}

Розглянемо цю програму докладніше. Рядок

template <class X> void swapargs(X &а, X &b)

повідомляє компілятор, що: по-перше, створюється шаблон, і, по-друге, починається опис узагальненої функції. Тут параметр X задає узагальнений тип, який згодом буде замінений фактичним типом. Після цього рядка оголошується функція swapargs(), у якій змінні, що підлягають перестановці, мають узагальнений тип X. У функції main() функція swapargs Про викликається для трьох різних типів даних: int, double і char. Оскільки функція swapargs () є узагальненою, компілятор автоматично створює три її версії: для перестановки цілих і дійсних чисел, а також символів.

З шаблонами пов'язано декілька понять. По-перше, узагальнена функція. (тобто функція, оголошена за допомогою ключового слова template) називається також шаблонною функцією (template function). Ці терміни є синонімами. Конкретна версія узагальненої функції, що створюється компілятором, називається спеціалізацією (specialization) або функцією, що генерується (generated function). Процес генерації конкретної функції називається конкретизацією (instantiation) . Іншими словами, функція, що генерується, є конкретним екземпляром узагальненої функції.

Оскільки мову C++ не рахує кінець рядка символом кінця оператора, розділ template у визначенні узагальненої функції не зобов'язаний знаходитися в одному рядку з ім'ям функції. Цю особливість ілюструє наступний приклад.

 

template <class X>

void swapargs(X &a, X &b)

{

X temp;

temp = а ; а = b; b = temp;

}

В цьому випадку слід пам'ятати, що між оператором template і початком визначення узагальненої функції не повинно бути інших операторів. Наприклад, наступний фрагмент помилковий.

 

// Цей фрагмент містить помилку.

template <class X>

int i; // Помилка

void swapargs(X &a, X &b)

{

X temp;

temp = а; а = b; b = temp;

}

 

Як випливає з коментаря, специфікація template повинна безпосередньо передувати визначенню функції.

Функція з двома узагальненими типами

Використовуючи список, елементи якого розділені комами, можна визначити декілька узагальнених типів даних в операторові template. Наприклад, в наступній програмі створюється шаблонна функція, що має два узагальнені типи.

#include <iostream>

using namespace std;

template <class typel, class type2>

void myfunc(typel x, type2 у)

{

cout « x « ' ' « у << ' \n';

}

int main()

{

myfunc(10, "Я люблю C++");

myfunc(98.6, 19L);

return 0;

}

При генерації конкретних екземплярів функції myfunc() компілятор замінює шаблонні типи typel і type2 типами int і char*, а також double і lone відповідно.

При створенні шаблонної функції компілятор може автоматично генерувати стільки її різних варіантів, скільки існує способів виклику цієї функції в програмі.

Явне перевантаження узагальненої функції

Не дивлячись на те що узагальнена функція перенавантажує сама себе, її можна перенавантажувати явно. Цей процес називається явною спеціалізацією (explicit specialization). Переобтяжена функція заміщає (або "маскує") узагальнену функцію, пов'язану з даною конкретною версією. Розглянемо модифіковану версію програми, призначеної для перестановки два змінних.

 

// Заміщення шаблонної функції.

#include <iostream>

using namespace std;

template <class X> void swapargs(X &a, X &b)

{

X temp;

temp = а; а = b; b = temp;

cout « "Усередині функції swapargs.\n";

}

// Ця функція заміщає узагальнену версію функції

// swapargs(), призначену для перестановки цілих чисел.

void swapargs(int &a, int &b)

{

int temp;

temp = а; а = b;

b = temp;

cout « "Усередині спеціалізації функції swapargs для цілих чисел.\n";

}

 

int main() {

int i=10, j=20;

double x=10.1, y=23.3;

char a='x', b= ' z ' ;

cout << “Вихідні значення i, j: ” << i << ‘ ’ << j << ‘\n’;

cout << “Вихідні значення x, y: ” << x << ‘ ’ << y << ‘\n’;

cout << “Вихідні значення a, b: ” << a << ‘ ’ << b << ‘\n’;

swapargs(i, j); // Виклик явно переобтяженої функції swapargs().

swapargs(х, у); // Виклик узагальненої функції swapargs().

swapargs(а, b); // Виклик узагальненої функції swapargs()

 

cout << “Переставлені значення i, j: ” << i << ‘ ’ << j << ‘\n’;

cout << “Переставлені значення x, y: ” << x << ‘ ’ << y << ‘\n’;

cout << “Переставлені значення a, b: ” << a << ‘ ’ << b << ‘\n’;

return 0;

}

 

Як вказано в коментарях програми, при виклику функції swapargs() активізується її явно переобтяжена версія. Таким чином, компілятор не генерує версію шаблонної функції swapargs ().

Недавно з'явилася нова синтаксична конструкція, призначена для позначення явної спеціалізації функції. Цей метод використовує ключове слово template. Наприклад, переобтяжену функцію swapargs() з попереднього прикладу можна переписати таким чином.

// Нова синтаксична конструкція для спеціалізації

template<> void swapargs<int>(int &a, int &b)

{

int temp;

temp = а;

а = b;

b = temp;

cout << "Усередині спеціалізації  swapargs для цілих чисел.\n";

}

Як видимий, новий спосіб визначення спеціалізації містить конструкцію template<>. Тип даних, для якого призначена спеціалізація, указується усередині кутових дужок після імені функції. Для спеціалізації будь-якого іншого типу узагальненої функції використовується така ж синтаксична конструкція. В даний час обидва способи визначення спеціалізації еквівалентні. Ймовірно, в майбутньому новий стиль може виявитися переважнішим.

За допомогою явної спеціалізації шаблону можна настроювати версію узагальненої функції на конкретну ситуацію. Це дозволяє максимально ефективно використовувати її для заданого типу даних. Проте, як правило, якщо для різних типів даних потрібні різні версії функції, слід застосовувати перевантаження, а не шаблони.


 

А также другие работы, которые могут Вас заинтересовать

62434. Социальная структура общества, социальная структура как внутреннее устройство общества 390.66 KB
  Понятие социальной структуры общества Социальные общности и их типологизация Социальные группы и их типологизация Социально классовая структура общества и социальная стратификация Государственная социальная поддержка населения – важнейшее направление политики белорусского государства...
62435. Основы цветоведения 277.74 KB
  В цветоведении есть раздел посвященный физическим свойствам цвета. Основных цвета три: красный синий жёлтый Основные краски нельзя получить механическим смешиванием. Три основные краски при смешивании между собой дают все остальные цвета...
62437. Логические основы обработки информации 465.89 KB
  Мышление изучают и психология, и педагогика, и многие другие науки. По содержанию человеческое мышление бесконечно многообразно,ведь думать можно о чём угодно. Но мысли возникают и строятся по одним и тем же законам, подчиняются одним и тем же принципам, имеют одни и те же схемы или формы.
62438. СОЦИАЛЬНЫЙ КОНФЛИКТ 17.32 KB
  Большая группа социологов полагающая что конфликт является неотъемлемой частью бытия главным двигателем общественного развития вычленяют функции конфликта которые по их мнению благотворно сказываются на актуальном состоянии общества и способствуют его развитию а именно...
62439. Субъекты предпринимательской деятельности 44.91 KB
  Субъекты предпринимательской деятельности 1. Правовой статус субъекта предпринимательской деятельности признаки. Способы образования субъектов предпринимательской деятельности.
62440. Этнические общности и их формы. Основные признаки нации. Белорусы как нация 221.78 KB
  Цели занятия: Обучающие: формирование знаний учащихся об этнонациональной структуре общества основных этапах развития этноса Воспитательные: формирование гражданственности патриотизма и интернационализма национальной и религиозной терпимости и уважения к традициям и культуре других народов...