Программирование >>  Обобщенные обратные вызовы 

1 ... 56 57 58 [ 59 ] 60 61 62 ... 84


з. Что следует изменить в программе, чтобы она работала так, как, вероятно, ожидает программист?

Все, что нам надо, - внести в исходный текст такие изменения, чтобы компилятор не мог рассматривать его как объявление функции. Есть два способа сделать это. Вот более привлекательный способ,

пример 29-3(а): устранение неоднозначности при помощи дополнительной пары скобок (неплохое

решение; оценка 7/10)

deque<strnng> col 13((istream iterator<string>(cin)), istream iterator<string>() );

Здесь мы просто добавили дополнительные скобки вокруг параметра, давая понять компилятору, что мы хотим, чтобы это был параметр конструктора, а не объявление параметра функции. Этот способ срабатывает, поскольку хотя istream iterator<string>(cin) может быть объявлением переменной (или, как уже упоми-налось, объявлением параметра), (istream iterator<string>(cin)) объявлением параметра быть не может. Соответственно, исходный текст примера 29-Зя не может быть объявлением функции по той же причине, по которой не может быть объявлением функции void f ( (int i) ), a именно - из-за наличия дополнительных скобок, которые не могут окружать параметр в объявлении функции.

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

> Рекомендация

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

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

и, кроме того, облегчающий чтение и понимание текста человеком.

Пример 29-3(6): использование именованных переменных (рекомендуемое решение; оценка 10/10)

istream iterator<string> fi rst( cin ), last; deque<string> coll3( first, last );

Как в примере 29-Зя, так и в при.мерс 29-36 достаточно внести предлагаемые изменения только в один параметр, но чтобы быть последовательным, я сделал именованными оба параметра.

> Рекомендация

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



Резюме

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

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



Задача 30. Двойная точность -

вежливость программистов Сложность: 4

Нет, эта задача не об этике. А о том, что есть разные виды плавающей точки . Давайте проверим, насколько хорошо вы разбираетесь в основных операциях с плавающей точкой в С и С++.

Вопрос для новичка

1. В чем заключается различие между float и double? Вопрос для профессионала

2. Пусть приведенная далее программа выполняется одну секунду (что достаточно необычно для современных настольных компьютеров).

int main О {

double x = 1е8;

whileC x > О ) { --х;

Как вы думаете, сколько времени будет выполняться этот код, если изменить dou-ble на float? Почему?

Решение

Два слова о float и double

1. В чем заключается различие между float и double?

Процитируем отрывок из стандарта С++:

Имеется три типа с плавающей точкой: float, double и long double. Тип double обеспечивает, как минимум, ту же точность, что и float, а тип long double - как минимум, ту же точность, что и doub 1е. Множество значений, которые принимает тип float, представляет собой подмножество значений типа double; а множество значений типа double является подмножеством значений типа long double. - [С++03, §3.9.1/8]

Как это определение, в особенности его последнее предложение, могут повлиять на ваши исходные тексты?

Колесо времени

2. Пусть приведенная далее программа выполняется одну секунду (что достаточно необычно для современных настольных компьютеров).

int main С) {

double x = leS; whileC X > О ) {

Как вы думаете, сколько времени будет выполняться этот код, если изменить double на float? Почему?



1 ... 56 57 58 [ 59 ] 60 61 62 ... 84

© 2006 - 2025 pmbk.ru. Генерация страницы: 0.001
При копировании материалов приветствуются ссылки.
Яндекс.Метрика