logo
Харви Дейтел, Пол Дейтел Как программировать на С++ / 02-Deitel-Стр-115-214

126 Глава 2

Типичная ошибка программирования 2.16

Забывают включить файл math.h в программы, использующие библиотеку матема -тических функций.

Обратите внимание, что мы объявили переменные amount, principal и rate типа double. Мы сделали это для простоты, поскольку мы имеем дело с дробной частью долларов и нам нужен тип с плавающей запятой. К со­жалению, это может вызвать затруднения. Поясним, к каким недоразуме­ниям может приводить использование float или double при представлении суммы долларов (в предположении, что печать осуществляется с setpreci-sion(2)). Две суммы долларов типа float, хранящиеся в памяти, могут иметь значения 14.234 (печатается как 14.23) и 18.673 (печатается как 18.67). Когда эти суммы складываются, результат внутри компьютера получается 32.907, что будет напечатано как 32.91. Таким образом, печать может иметь вид:

14.23 + 18.67

32.91

но пользователь, складывающий эти числа в том виде, в котором они напеча­таны, ожидает получить сумму 32.90!

Хороший стиль программирования 2.21

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

Оператор вывода

cout « setw(3) « year

« setiosflags(ios::fixed | ios::showpoint)

« setw(21) « setprecision{2) « amount « endl;}

печатает значения переменных year и amount в формате, заданном парамет­ризованными манипуляторами потока setw, setiosflags и setprecision. Обраще­ние setw(3) определяет, что следующая выходная величина будет напечатана с шириной (размером) поля 3, т.е. ее значение будет содержать по крайней мере 3 символьных позиции. Если длина выходной величины менее 3 симво­лов, она по умолчанию будет выравнена в поле по правому символу. Если длина выходной величины более 3, размер поля будет увеличен, чтобы вместить полную величину. Можно использовать обращение setiosflags(ios::left) для того, чтобы задать выравнивание по левому символу.

Остальная часть задания формата в приведенном операторе вывода ука­зывает, что переменная amount должна печататься как значение с фикси­рованной точкой и с обязательной печатью этой точки (задается выражением setiosflags(ios::fixed | ios::showpoint)) с правым выравниванием в поле разме­ром в 21 символ (задается выражением setw(21)) и с двумя цифрами после десятичной точки (задается выражением setprecision(2)). Мы подробнее рас-

Управляющие структуры 127

смотрим богатые возможности С++ по форматированию ввода-вывода в главе 11.

Заметьте, что вычисление выражения 1.0 + rate, являющегося аргумен­том функции pow, производится в теле оператора for. В действительности вычисление этого выражения дает одинаковый результат в каждом цикле, так что его повторные вычисления расточительны.

Совет по повышению эффективности 2.3

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