Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
http.doc
Скачиваний:
0
Добавлен:
01.05.2025
Размер:
1.34 Mб
Скачать

Указатели на указатели - многочисленное перенаправление

Концепция массивов указателей открыта и проста, поскольку индексы имеют вполне определенное значение. Тем не менее, в случае, когда один указатель указывает на другой, могут возникнуть проблемы. Указатель на указатель является формой многочисленного перенаправления или цепочки указателей. Рассмотрим рис.

Рисунок: Одиночное и многочисленное перенаправление

В случае обычных указателей, указатель содержит адрес некоторого участка памяти, содержащего некоторое значение. В случае указателя на указатель, первый указатель содержит адрес второго, который в свою очередь содержит адрес участка памяти, содержащего некоторое значение.

Многочисленное перенаправление может и дальше расширяться. Но существует немного случаев, когда необходимо что-то более мощное, чем указатель на указатель. Излишнее перенаправленние приводит к концептуальным ошибкам, которые очень трудно исправлять. (Не надо путать многочисленное перенаправление со связанными списками, которые используются в базах данных.)

Переменная, являющаяся указателем на указателе должна быть описана следующим образом. Это выполняется путем помещения двух звездочек перед именем. Например, следующее объявление сообщается компилятору, что newbalance - это указатель на указатель типа float: float **newbalance; Важно понимать, что newbalance - это не указатель на число с плавающей точкой, а указатель на указатель на вещественное число.

Для получения доступа к целевому значению, косвенно указываемому указателем на указатель, следует применить оператор * два раза, как показано в следующем примере: #include <stdio.h> int main(void) { int x, *p, **q; x = 10; p = &x; q = &p; printf ("%d", **q) ; /* вывод значения x */ return 0; } Здесь p объявляется как указатель на целое, a q - это указатель на указатель на целое. Вызов printf() выводит число 10 на экран.

Инициализация указателей

После объявления указателя и до первого присвоения ему значения, указатель может содержать неизвестное значение. Если попытаться использовать указатель до сообщения ему значения, можно нарушить работу не только программы, но и всей операционной системы.

По существующим соглашениям неиспользуемый указатель должен содержать нулевое значение. То, что указатель имеет нулевое значение, не обеспечивает безопасности. Если использовать нулевой указатель слева от оператора присваивания, это может привести к «зависанию» программы.

Поскольку предполагается, что нулевой указатель не используется, можно его использовать для упрощения большинства подпрограмм, работающих с указателями. Например, можно использовать нулевой указатель для пометки конца массива указателей. Если это сделано, то процедура, осуществляющая доступ к массиву, будет знать, что она достигла конца массива при обнаружении нулевого значения. Данный способ реализации функции search() показан ниже: /* Ищем имя */ int search(char *р[], char *name) { register int t; for(t=0; p[t]; ++t) if(!strcmp(p[t], name)) return t; return -1; /* не найдено */ } Цикл for в search() запускается каждый раз при обнаружении нулевого указателя. Поскольку конец массива помечен нулем, условие, управляющее циклом, останавливает цикл (выдает ложное состояние) при достижении конца массива.

При профессиональном написании программ типично инициализировать строки. Пример этого был показан в функции serror(). Ниже приводится другой вариант объявления строк: char *р = "hello world\n"; Как можно видеть, указатель р - это не массив. Все компиляторы С создают так называемую таблицу строк, используемую компилятором для хранения строковых констант, используемых программой. Следовательно, данный оператор объявления помещает адрес "hello world" в указатель р. р может быть использован в программе как обычная строка. Например, следующая программа абсолютно корректна: #include <stdio.h> #include <string.h> char *p = "hello world"; int main(void) { register int t; /* вывод строки в прямом и обратном порядке */ printf(р); for(t=strlen(р)-1; t>-1; t--) printf ("%c", p[t]); return 0; }

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]