Г. Шилтд - Самоучитель C++ (DJVU) (1114955), страница 26
Текст из файла (страница 26)
При этом компилятор выставит флаг ошибки даже раньше того, как встретится такая инструкция, поскольку сама перегрузка этих двух функций внутренне неоднозначна, и компилятор не будет знать, какую из них предпочесть. 4. Другим видом неоднозначности при перегрузке функций является случай, когда одна или более перегруженных функций используют аргумент по умолчанию.
Рассмотрим программу: 1бб Самоучитель С++ О Неоднозначность, основанная на аргументах по умолчанию Д,1 и перегрузке Функций ((ьпс1цс(е <1овсвею> цв1пя пагпсврасс вгс(; 1п( Е (1пс а) гесцгп а * а; (пг г"(1п( а, (пг Ь =- О) гегцгп а " Ь; ни амато () соцс « Х(10, 2); // вызывается 1(1пе, гпс) сонг « г(10)г // неоднозначность, что вьзвать т(1ПВ, гпс) или Г(1Пс) Эз ? егитп 0; Здесь вызов функции 1(1(), 2) совершенно правилен и не ведет к неоднозначности.
Однако у компилятора нет способа выяснить, какую версию функции Г() вызывает версия т(10) — первую или вторую, в которой параметр Ь передается по умолчанию. Попытайтесь провести компиляцию всех предыдущих программ, в которых имеет место неоднозначность. Запомните сообщения об ошибках. Это поможет вам сразу распознать ошибки неоднозначности, если они появятся в ваших программах.
5.6. Определение адреса перегруженной Функции В заключение этой главы вы узнаете, как найти адрес перегруженной функции. Так же, как и в С, вы можете присвоить адрес функции указателю и получить доступ к функции через этот указатель. Адрес функции можно найти„если поместить имя функции в правой части инструкции присваивания без всяких скобок или аргументов. Например, если гарΠ— это функ- Глава 5.
Перегрузка функций ция, причем правильно объявленная, то корректным способом присвоить переменной р адрес функции каро является инструкция: р — кар; В языке С любой тип указателя может использоваться как указатель на функцию, поскольку имеется только одна функция, на которую он может ссылаться. Однако в С++ ситуация несколько более сложная, поскольку функция может быть перегружена. Таким образом, должен быть некий механизм, который позволял бы определять адреса перегруженных версий функции. Решение оказывается не только элегантным, но и эффектным. Сг)особ обьявления указателя и определяет то, адрес какой из перегруженных версий функции будет получен.
Уточним, объявления указателей соответствуют объявлениям перегруженных функций. Функция, объявлению которой соответствует объявление указателя, и является искомой функцией. ) Здесь представлена программа, которая содержит две версии функции арасе(). Первая версия выводит на экран некоторое число пробелов, заданное в переменной соуп(. Вторая версия выводит на экран некоторое число каких-то иных символов, вид которых задан в переменной с)). В функции и) а)по объявляются оба указателя на эти функции. Первый задается как указатель на функцию, имеющую только один целый параметр. Второй объявляется как указатель на функцию, имеющую два параметра.
/* Иллюстрация присваивания и получения указателей на перегруженные функции ~/ ((гпсапс)е <гоаЕгеал> пз1по патеврасе вес) г вывод заданного в переменной соппс числа пробелов лото врасе(хпь соппс) гог(; соппв; сопле --) сопл « вывод заданного в переменной сопаЬ числа символов, вид которых задан в переменной сЬ чотб врасе('пп соппг, сЬаг сЬ) Гог(; соппс; сеанс --) сов « сЬ; Самоучитель (п( гпа)п С) 1* Создание укаяаталя на функцию с одним целым параметром.
*1 чоЫ (*Гр1» (впав): !* Создание указателя на функцию с одним целым и одним символьным параметром. *1 чо1с) (*йр2) (ъпд, сиад); тр1 = ярасе; /1 получение адреса функции ярасе (ъпТ) гр2 = ярасе; !! получение адреса функции арасе (фпад, сбат) гр1 (22); !! выводит 22 пробела соцд « ")хп"; ~р2(30, 'и'); !1 выводит ЗО символов х соцб « "(1п"; тебцтп 0; Как показано в комментариях, на основе того, каким образом объявляются указатели ф! и ~р2, компилятор способен определить, какой из них на какую из перегруженных функций будет ссылаться.
Повторим, если вы присваиваете адрес перегруженной функции указателю на функцию, то объявление указателя определяет, адрес какой именно функции ему присваивается. Более того, объявление указателя на функцию должно точно соответствовать одной и только одной перегруженной функции. Если это не так, будет внесена неоднозначность, что приведет к ошибке при компиляции программы. 1. Ниже приведены две перегруженные функции. Покажите, как получить адрес каждой из них. )п( <(1й (1пб а, )п( Ь) гетц и а — Ь; ) 11оаб с)1Е (Х1оас а, й1оаб Ы ( леСцтп а — Ь; Глава а.
Перегрузка функций Теперь вам необходимо выполнить следующие упражнения и ответить на вопросы: 1. Перегрузите конструктор йа1еО из раздела 5.1, пример 3 так, чтобы он имел параметр типа Ове 1. (Вспомните, что Мите 1 — это тип данных, определенный стандартными библиотечными функциями времени и даты компилятора С++.) 2. Что неправильно в следующем фрагменте? с1ааа аатр ( )ггГ а; рсЫ гс: затр (айаг 1) ( а /у )п$ гаага () ( яаагр х, у(10) 1 3. 5.
6. 7. 8. Приведите два довода в пользу того, почему вам могло бы потребоваться перегрузить конструктор класса. Какова основная форма конструктора копий? Какой тип операций ведет к вызову конструктора копий? Кратко объясните, зачем нужно ключевое слово оуег1оае, и почему оно больше не употребляется. Объясните, что такое аргумент по умолчанию? Создайте функцию гетегяеО с двумя параметрами. Первый параметр я1т — это указатель на строку, порядок следования символов в которой, после возвращения функцией своего значения, должен быть заменен на обратный.
Второй параметр соип$ задает количество переставляемых в строке а(г символов. Значение совп1 по умолчанию должно быть таким, чтобы в случае его задания функция геуегаео меняла порядок следова- ния символов в целой строке. Что неправильно в следующем прототипе функции? сваг ис г)и ар(сваг "зсг, )пГ аде = О, сааг сй) Самоучитель С++ 1О. Приведите несколько причин появления неоднозначности при пере- грузке функций. 11, Что неправильно в следуюшем фрагменте? чоЫ сошрове (с(ооЬ1е тотс, 1п( с)1ч1еот = т) ' чо1с) ссяприсе (с)оо)оте *сап); // соп~рисе(ьх); 12.
При присваивании указателю адреса перегруженной функции, что опре- деляет конкретную версию используемой функции? В этом разделе проверяется, хорошо ли вы усвоили материал этой и предыдуших глав. 1. Создайте функцию огдег0, которая получает два параметра-ссылки на целые. Если первый аргумент больше второго, поменяйте их значения. В противном случае ничего делать не надо. Таким образом, порядок следования двух аргументов, используемых при вызове функции огйег0, должен быть таким, чтобы всегда после возвращения функцией своего значения первый аргумент был меньше второго. Например, если дано )осх=(,у=о; осс(ее [х, у); то после вызова функции х будет равен О, а у будет равен 1.
2. Почему следующие две перегруженные функции внутренне неоднозначны? (от Е(~ос. а); (и( с (ьпс ьа); 3. Объясните, почему использование аргумента по умолчанию связано с перегрузкой функций. 4. Пусть дано следующее неполное описание класса, добавьте конструкторы так, чтобы оба объявления в функции в)а1п0 были правильны. (Подсказка: вам необходимо дважды перегрузить конструктор вагяр0.? стаее еатпр ( тес а; Глава 6 Введение в перегрузку операторов В этой главе рассматривается очередное важное свойство С++: перегрузка операторов.
Это свойство позволяет определять значение операторов С++ относительно задаваемых вами классов. Путем перегрузки связанных с классами операторов можно легко добавлять в программу новые типы данных. Перед тем как продолжить, необходимо правильно ответить на следующие вопросы и сделать упражнения. 1. Покажите, как перегрузить конструктор для следующего класса так, чтобы можно было создавать не только инициализируемые, но и неинициализируемые объекты. 1При создании таких объектов присвойте переменным х и у значение О.) с1аав п~ус1авв ( »«»С к,у; рыб»с: тус1азз(же», »ле 5( ( х=1; у= э'; > «'/ 2. Используя класс из вопроса 1, покажите, как с помощью аргументов по умолчанию можно избежать перегрузки конструктора шусЬааО. 3. Что неправильно в следующем объявлении? (п( Г(»г»Г.
а = О, до»»Ые Ьа1апсе«; 4. Что неправильно в следующих двух перегруженных функциях? »'оЫ г" (1пс а); » оЫ 1(ъг»е аа«; 5. Когда удобнее использовать аргументы по умолчанию? Когда этого луч- ще не делать? Самоучитель С++ в. Дано следующее определение класса. Возможно ли динамически выделить память для массива объектов такого типа? с1аяя сеяя ( с'пат *р; 3.ГЙ. *ч; Гпя сопп'с' рио1ъс: яеят.(спаг *х, Гпя *у, Гпя с) ( р = х„ ст— у: сопя с = с; уу 7. Что такое конструктор копий и при каких условиях он вызывается? 6.1.
Основы перегрузки операторов Перегрузка операторов напоминает перегрузку функций. Более того, перегрузка операторов является фактически одним из видов перегрузки функций. Однако при этом вводятся некоторые дополнительные правила. Например, оператор всегда перегружается относительно определенного пользователем типа данных, такого, как класс, Другие отличия будут обсуждаться ниже по мере необходимости.
Когда оператор перегружается, то ничего из его исходного значения не теряется. Наоборот, он приобретает дополнительное значение, связанное с классом, для которого оператор был определен. Для перегрузки оператора создается онератнор-Функция (орега1ог Янсйоп). Чаще всего, оператор-функция является членом класса или дружественной классу, для которого она определена. Однако есть небольшая разница между оператор-функцией — членом класса и дружественной оператор-функцией.