h
не указан тип аргумента. Это не означает, что функция h
не получает ни одного аргумента; это значит: принимает любой набор аргументов и надеется, что это набор при вызове окажется правильным. И снова отметим, что хороший компилятор предупредит об этой проблеме, а программа lint
перехватит ее.
Существует специальный набор правил, регламентирующих преобразование аргументов, если в области видимости нет прототипа функции. Например, переменные типов char
и short
преобразуются в переменные типа int
, а переменные типа float
— в переменные типа double
. Если вы хотите знать, скажем, что произойдет с переменной типа long
, загляните в хороший учебник по языку С. Наша рекомендация проста: не вызывайте функций, не имеющих прототипов.
Обратите внимание на то, что, хотя компилятор допускает передачу аргументов неправильного типа, например параметр типа char*
вместо параметра типа int, использование таких аргументов приводит к ошибкам. Как сказал Деннис Ритчи: “С — это язык программирования со строгим контролем типов и слабой проверкой”.
27.2.3. Определения функций
Можете определять функции точно так же, как в языке С++. Эти определения являются прототипами функций.
double square(double d)
{
return d*d;
}
void ff
{
double x = square(2); /* OK: переводим 2 в 2.0 и вызываем */
double y = square; /* пропущен аргумент */
double y = square("Hello"); /* ошибка: неправильный тип
аргументов */
double y = square(2,3); /* ошибка: слишком много аргументов */
}
Определение функции без аргументов не является прототипом функции.
void f { /* что-то делает */ }
void g
{
f(2); /* OK в языке C; ошибка в языке C++ */
}
Код
void f; /* не указан тип аргумента */
означающий, что функция f
может принять любое количество аргументов любого типа, выглядит действительно странно. В ответ на это я изобрел новое обозначение, в котором понятие “ничего” указывалось явным образом с помощью ключевого слова void
(
void f(void); /* не принимает никаких аргументов */
int old_style(p,b,x) char* p; char b;
{
/* ... */
}
int
. Итак, параметр x
является аргументом функции old_style
, имеющим тип int
. Мы можем вызвать функцию old_style
следующим образом:
old_style; /* OK: пропущены все аргументы */
old_style("hello",'a',17); /* OK: все аргументы имеют правильный тип */
old_style(12,13,14); /* OK: 12 — неправильный тип */
/* но old_style может не использовать p */
Компилятор должен пропустить эти вызовы (но мы надеемся, что он предупредит о первом и третьем аргументах).
Мы рекомендуем придерживаться следующих правил проверки типов аргументов функций.
• Последовательно используйте прототипы функций (используйте заголовочные файлы).
• Установите уровень предупреждений компилятора так, чтобы перехватывать ошибки, связанные с типами аргументов.
• Используйте (какую-нибудь) программу lint
.
В результате вы получите код, который одновременно будет кодом на языке C++.
27.2.4. Вызов функций, написанных на языке С, из программы на языке С++, и наоборот