
РИС. 13.2. Инициализация массива.
Отметим, что в нашей программе массиву nameзадан размер:
char name [81];
Поскольку массив nameдолжен читаться во время работы программы, у компилятора нет другого способа узнать заранее, сколько памяти нужно выделить для массива. Это нс символьная константа, в которой компилятор может посчитать символы. Поэтому мы предположили, что 80 символов будет достаточно, чтобы поместить в массив фамилию пользователя.
Обычно бывает удобно иметь массив символьных строк. В этом случае можно использовать индекс для доступа к нескольким разным строкам. Покажем это на примере:
static char *mytal[LIM] = {"Быстро складываю числа",
"Точно умножаю",
"Записываю данные",
"Правильно выполняю команды",
"Понимаю язык Си"};
Разберемся в этом описании. Вспомним, что LIMимеет значение 5, мы можем сказать, что mytalявляется массивом, состоящим из пяти указателей на символьные строки. Каждая строка символов, конечно же, представляет собой символьный массив, поэтому у нас есть пять указателей на массивы. Первым указателем является mytal[0], и он ссылается на первую строку. Второй указатель mytal[1]ссылается на вторую строку. Каждый указатель, в частности, ссылается на первый символ своей строки:
*mytal[0] == 'Б', *mytal[1] == 'Т', mytal[2] == 'З'
и т. д.
Инициализация выполняется по правилам, определенным для массивов. Тексты в кавычках эквивалентны скобочной записи
{{...}, {...}, ..., {...}};
где многоточия подразумевают тексты, которые мы поленились напечатать. В первую очередь мы хотим отметить, что первая последовательность, заключенная в двойные кавычки, соответствует первым парным скобкам и используется для инициализации первого указателя символьной строки. Следующая последовательность в двойных кавычках инициализирует второй указатель и т. д. Запятая разделяет соседние последовательности.
Кроме того, мы могли бы явно задавать размер строк символов, используя описание, подобное такому:
static char mytal[LIM][LINLIM];
Разница заключается в том, что второй индекс задает "прямоугольный" массив, в котором все "ряды" (строки) имеют одинаковую длину. Описание
static char *mytal [LIM]
однако, определяет "рваный" массив, где длина каждого "ряда" определяется той строкой, которая этот "ряд" инициализировала. Рваный массив не тратит память напрасно.

PИС. 13.3. Прямоугольный массив или pваный
Возможно, вы заметили периодическое упоминание указателей в нашем рассказе о строках. Большинство операции языка Си, имеющих дело со строками, работает с указателями. Например, рассморим приведенную ниже бесполезную, но поучительную программу
/* указатели и строки */
#define PX(X) printf("X = %s; значение = %u; &X = %u\n", X, X, &X)
main( ) {
static char *mesg = "He делай глупостей!";
static char *copy;
copy = mesg;
printf(" %s \n" , copy);
PX(mesg);
PX(copy);
}
Взглянув на эту программу, вы можете подумать, что она копирует строку "Не делай глупостей!", и при беглом взгляде на вывод вам может показаться правильным это предположение:
He делай глупостей!
mesg = He делай глупостей!; значение = 14; &mesg = 32
copy = He делай глупостей!; значение = 14; &сору = 34
Но изучим вывод РХ(). Сначала X, который последовательно является mesgи сору, печатается как строка ( %s). Здесь нет сюрприза. Все строки содержат "Не делай глупостей!".
Далее ... вернемся к этому несколько позднее.
Третьим элементом в каждой строке является &X, т. е. адрес X. Указатели mesgи copyзаписаны в ячейках 32 и 34 соответственно.
Теперь о втором элементе, который мы называем значением . Это сам X. Значением указателя является адрес, который он содержит. Мы видим, что mesgссылается на ячейку 14, и поэтому выполняется сору.
Смысл заключается в том, что сама строка никогда не копируется. Оператор copy=mesg;создаст второй указатель, ссылающийся на ту же самую строку.
Зачем все эти предосторожности? Почему бы не скопировать всю строку? Хороню, а что эффективнее - копировать один адрес или, скажем, 50 отдельных элементов ? Часто бывает, что адрес это все, что необходимо для выполнения работы.
Читать дальше