Эта функция также имеет доступ к названиям учреждений, хотя их не использует. Заметим, что мы должны применять операцию &для получения адреса структуры. В отличие от имени массива имя структуры само по себе нe является синонимом своего адреса.
Наш следующий способ применим к массивам структур и является вариантом данного способа.
Предположим, у нас есть массив структур. Имя массива является синонимом его адреса, поэтому его можно передать функции. С другой стороны, функции будет необходим доступ к структурному шаблону. Чтобы показать, как такая программа работает (рис. 14.8), мы расширим нашу программу таким образом, чтобы она обслуживала двух человек, а мы, следовательно, имели бы массив двух структур funds.
/* передача массива структур в функцию */
struct funds {
char *bank;
float bankfund;
char *save;
float savefund; }
jones[2] ={
{ " Банк синьора Помидора" ,
1023.43,
" Сбережения и займы Снупи" ,
4239.87 },
{ " Банк Честного Джека",
976.57,
"Накопления по предварительному плану",
1760.13 } };
main( )
{
float total, sum( );
printf("Джонсы имеют всего %.2f долл.\n", sum(jones));
}
float sum(money);
struct funds *money;
{
float total;
int i;
for( i = 0, total = 0; i < 2; i++, money++ )
total += money- > bankfund + money -> savefund;
return(total);
}
РИС. 14. 8. Программа, передающая функции массив структур.
Программа выдает
Джонсы имеют всего 8000.00 долл.
(Что за круглая сумма! Можно подумать, что эти цифры взяты с потолка.) Имя массива jonesявляется указателем на массив. В частности, оно ссылается на первый элемент массива, который является структурой jones[0]. Таким образом, в начале указатель money за дается через
money = &jones[0];
Затем использование операции - >позволяет нам добавить два вклада для первого Джонса. Это действительно очень похоже на последний пример. Далее, цикл forувеличивает указатель moneyна 1. Теперь он ссылается на следующую структуру, jones[1], и остаток вкладов может быть добавлен к total.
Вот два основных замечания:
1. Имя массива можно использовать для передачи в функцию указателя на первую структуру в массиве.
2. Затем можно использовать арифметическую операцию над указателем, чтобы передвигать его на последующие структуры в массиве. Заметим, что вызов функции
sum(&jones[0])
дал бы тот же самый эффект, что и применение имени массива, так как оба они ссылаются на один и тот же адрес. Использование имени массива является просто косвенным способом передачи адреса структуры.
Мы не будем больше рассказывать о структурах, но хотелось бы отметить одно очень важное использование структур: создание новых типов данных. Пользователи компьютеров разработали новые типы данных, гораздо более эффективные для определенных задач, чем массивы и простые структуры, которые мы описали.
Эти типы имеют такие названия, как очереди, двоичные деревья, неупорядоченные массивы, рандомизированные таблицы и графы. Многие из этих типов создаются из "связанных" структур. Обычно каждая структура будет содержать один или два типа данных плюс один или два указателя на другие структуры такого же типа. Указатели служат для связи одной структуры с другой и для обеспечения пути, позволяющего вам вести поиск по всей структуре. Например, на рис. 14.9 показано двоичное дерево, в котором каждая отдельная структура (или "узел") связана с двумя, расположенными ниже.

РИС. 14.9. Структура двоичного дерева.
Является ли эта разветвленная конструкция более эффективной чем массив? Рассмотрим случай дерева с 10 уровнями узлов. Если вы составите его, то найдете 1023 узла, в которых вы можете запомнить, скажем, 1023 слова. Если слова упорядочены, согласно некоторому разумному плану, вы можете начать с верхнего уровня и находить любое слово в лучшем случае за 9 перемещений, если ваш поиск идет сверху вниз с одного уровня на следующий. Если слова находятся в массиве, вам, может быть, придется перебрать все 1023 элемента, прежде чем вы найдете нужное слово.
Когда вас интересуют более прогрессивные структуры данных, обратитесь к литературе по вычислительной технике. Используя структуры языка Си, вы сможете создавать типы, о которых вы прочитали.
Читать дальше