Программирование. Принципы и практика использования C++ Исправленное издание, стр. 63
error("неположительный второй аргумент функции area() \\ при вызове из функции framed_area()");int area2 = framed_area(1,z);if (y<=2 || z<=2) error("неположительный аргумент функции area()\\ при вызове из функции framed_area()");int area3 = framed_area(y,z);Это не только запутанно, но и неверно в принципе. Такой код можно написать, лишь точно зная, как функция
framed_area()area()Мы должны знать, что функция
framed_area()2framed_area()21В этом случае нам пришлось бы проверить каждый вызов функции
framed_area()framed_area()const int frame_width = 2;int framed_area(int x, int y) // вычисляем площадь, // ограниченную рамкой{ return area(x–frame_width,y–frame_width);}Это имя можно использовать в коде, вызывающем функцию
framed_area()if (1–frame_width<=0 || z–frame_width<=0) error("неположительный второй аргумент функции area() \\ при вызове из функции framed_area()");int area2 = framed_area(1,z);if (y–frame_width<=0 || z–frame_width<=0) error("неположительный аргумент функции area() \\ при вызове из функции framed_area()");int area3 = framed_area(y,z);Взгляните на этот код! Вы уверены, что он правильный? Он вам нравится? Легко ли его читать? Действительно, он уродлив (а значит, подвержен ошибкам). В результате наших неуклюжих попыток размер кода увеличился втрое, а детали реализации
framed_area()Существует более правильное решение!
Посмотрите на исходный код.
int area2 = framed_area(1,z);int area3 = framed_area(y,z);Он может быть неверным, но, по крайней мере, мы можем понять, что должно происходить. Мы можем сохранить эту ясность, поместив проверку ошибки в функцию
framed_area()5.5.2. Обработка ошибок в вызываемом модуле
Проверка корректности аргументов в функцию
framed_area()error()int framed_area(int x, int y) // вычисляем площадь, ограниченную рамкой{ const int frame_width = 2; if (x–frame_width<=0 || y–frame_width<=0) error("неположительный аргумент функции area() \\ при вызове из функции framed_area()"); return area(x–frame_width,y–frame_width);}Это решение выглядит неплохо, и нам больше нет необходимости писать проверку для каждого вызова функции
framed_area()Отметим нечто интересное: мы почти бессознательно заменили подход “вызывающий модуль должен сам проверять аргументы” на подход “функция должна проверять свои собственные аргументы”. Одним из преимуществ второго подхода является то, что проверка аргументов осуществляется в единственном месте. Теперь необязательно просматривать вызовы функции по всей программе. Более того, проверка производится именно там, где эти аргументы используются, поэтому мы имеем всю информацию, необходимую для проверки.
Итак, применим найденное решение к функции
area()int area(int length, int width) // вычисляем площадь прямоугольника{ if (length<=0 || width <=0) error("неположительный аргумент area()"); return length*width;}Этот фрагмент будет перехватывать все ошибки, возникающие в модулях, вызывающих функцию
area()framed_area()Проверка аргументов в функции выглядит настолько простой, что становится непонятным, почему люди не проводят ее постоянно? Одна из причин — пренебрежение ошибками, вторая — неряшливость при написании программ, но существуют и более уважительные причины.
• Мы не можем модифицировать определение функции. Функция является частью библиотеки, поэтому ее невозможно изменить. Возможно, она будет использована другими людьми, не разделяющими вашего подхода к обработке ошибок. Возможно, она принадлежит кому-то еще, и вы не имеете доступ к ее исходному коду. Возможно, она включена в постоянно обновляющуюся библиотеку, так что если вы измените эту функцию, то будете вынуждены изменять ее в каждой новой версии.
• Вызываемая функция не знает, что делать при выявлении ошибки. Эта ситуация типична для библиотечных функций. Автор библиотеки может выявить ошибку, но только вы знаете, что в таком случае следует делать.