#include ‹stdio.h›
#define MAXLINE 100
/* примитивный калькулятор */
main()
{
double sum, atof (char[]);
char line[MAXLINE];
int getline (char line[], int max);
sum = 0;
while (getline(line, MAXLINE) › 0)
printf ("\t%g\n", sum += atof(line));
return 0;
}
В объявлении
double sum, atof (char[]);
говорится, что sum - переменная типа double , a atof - функция, которая принимает один аргумент типа char[] и возвращает результат типа double .
Объявление и определение функции atof должны соответствовать друг другу. Если в одном исходном файле сама функция atof и обращение к ней в main имеют разные типы, то это несоответствие будет зафиксировано компилятором как ошибка. Но если функция atof была скомпилирована отдельно (что более вероятно), то несоответствие типов не будет обнаружено, и atof возвратит значение типа double , которое функция main воспримет как int , что приведет к бессмысленному результату.
Это последнее утверждение, вероятно, вызовет у вас удивление, поскольку ранее говорилось о необходимости соответствия объявлений и определений. Причина несоответствия, возможно, будет следствием того, что вообще отсутствует прототип функции, и функция неявно объявляется при первом своем появлении в выражении, как, например, в
sum += atof(line);
Если в выражении встретилось имя, нигде ранее не объявленное, за которым следует открывающая скобка, то такое имя по контексту считается именем функции, возвращающей результат типа int ; при этом относительно ее аргументов ничего не предполагается. Если в объявлении функции аргументы не указаны, как в
double atof();
то и в этом случае считается, что ничего об аргументах atof не известно, и все проверки на соответствие ее параметров будут выключены. Предполагается, что такая специальная интерпретация пустого списка позволит новым компиляторам транслировать старые Си-программы. Но в новых программах пользоваться этим - не очень хорошая идея. Если у функции есть аргументы, опишите их, если их нет, используйте слово void .
Располагая соответствующим образом описанной функцией atof , мы можем написать функцию atoi , преобразующую строку символов в целое значение, следующим образом:
/* atoi: преобразование строки s в int с помощью atof */
int atoi (char s[])
{
double atof (char s[]);
return (int) atof (s);
}
Обратите внимание на вид объявления и инструкции return . Значение выражения в
return выражение ;
перед тем, как оно будет возвращено в качестве результата, приводится к типу функции. Следовательно, поскольку функция atoi возвращает значение int , результат вычисления atof типа double в инструкции return автоматически преобразуется в тип int . При преобразовании возможна потеря информации, и некоторые компиляторы предупреждают об этом. Оператор приведения явно указывает на необходимость преобразования типа и подавляет любое предупреждающее сообщение.
Упражнение 4.2. Дополните функцию atof таким образом, чтобы она справлялась с числами вида
123.45e-6
в которых после мантиссы может стоять e (или E) с последующим порядком (быть может, со знаком).
Программа на Си обычно оперирует с множеством внешних объектов: переменных и функций. Прилагательное "внешний" ( external) противоположно прилагательному "внутренний", которое относится к аргументам и переменным, определяемым внутри функций. Внешние переменные определяются вне функций и потенциально доступны для многих функций. Сами функции всегда являются внешними объектами, поскольку в Си запрещено определять функции внутри других функций. По умолчанию одинаковые внешние имена, используемые в разных файлах, относятся к одному и тому же внешнему объекту (функции). (В стандарте это называется редактированием внешних связей (линкованием) ( external linkage ).) В этом смысле внешние переменные похожи на области COMMON в Фортране и на переменные самого внешнего блока в Паскале. Позже мы покажем, как внешние функции и переменные сделать видимыми только внутри одного исходного файла.
Поскольку внешние переменные доступны всюду, их можно использовать в качестве связующих данных между функциями как альтернативу связей через аргументы и возвращаемые значения. Для любой функции внешняя переменная доступна по ее имени, если это имя было должным образом объявлено.
Читать дальше