Бьерн Страуструп.
Язык программирования С++
102
// table.c: таблица имен и функция поиска
#include "dc.h"
extern char* strcmp(const char*, const char*);
extern char* strcpy(char*, const char*);
extern int strlen(const char*);
const int TBLSZ = 23;
name* table[TBLSZ];
name* look(char* p, int ins) { /* ... */ }
Отметим, что раз строковые функции описаны в самом файле table.c, транслятор не может проверить
согласованность этих описаний по типам. Всегда лучше включить соответствующий заголовочный
файл, чем описывать в файле .c некоторое имя как extern. Это может привести к включению "слишком
многого", но такое включение нестрашно, поскольку не влияет на скорость выполнения программы и ее
размер, а программисту позволяет сэкономить время. Допустим, функция strlen() снова описывается в
приведенном ниже файле main.c. Это только лишний ввод символов и потенциальный источник ошибок,
т.к. транслятор не сможет обнаружить расхождения в двух описаниях strlen() (впрочем, это может
сделать редактор связей). Такой проблемы не возникло бы, если бы в файле dc.h содержались все
описания extern, как первоначально и предполагалось. Подобная небрежность присутствует в нашем
примере, поскольку она типична для программ на С. Она очень естественна для программиста, но часто
приводит к ошибкам и таким программам, которые трудно сопровождать. Итак, предупреждение
сделано!
Наконец, приведем файл main.c:
// main.c: инициализация, основной цикл, обработка ошибок
#include "dc.h"
double error(char* s) { /* ... */ }
extern int strlen(const char*);
int main(int argc, char* argv[]) { /* ... */ }
В одном важном случае заголовочные файлы вызывают большое неудобство. С помощью серии
заголовочных файлов и стандартной библиотеки расширяют возможности языка, вводя множество
типов (как общих, так и рассчитанных на конкретные приложения; см. главы 5-9). В таком случае текст
каждой единицы трансляции может начинаться тысячами строк заголовочных файлов. Содержимое
заголовочных файлов библиотеки, как правило, стабильно и меняется редко. Здесь очень пригодился
бы претранслятор, который обрабатывает его. По сути, нужен язык специального назначения со своим
транслятором. Но устоявшихся методов построения такого претранслятора пока нет.
Достарыңызбен бөлісу: