После выполнения оператора с модификатором fixed переменная р будет указы вать на начало массива символов, составляющих строку. Этот массив оканчивается символом конца строки, т.е. нулевым символом. Поэтому данное обстоятельство мож но использовать для проверки конца массива. В С/С++ строки реализуются в виде мас сивов, оканчивающихся символом конца строки, а следовательно, получив указатель типа char* на строку, ею можно манипулировать таким же образом, как и в C/C++.
Ниже приведена программа, демонстрирующая доступ к символьной строке по указателю типа char*. // Использовать модификатор fixed для получения // указателя на начало строки. using System; class FixedString { unsafe static void Main { string str = "это тест"; // Получить указатель р на начало строки str. fixed(char* р = str) { // Вывести содержимое строки str по указателю р. for(int i=0; p[i] != 0; i++) Console.Write(p[i]); } Console.WriteLine; } }
Эта программа дает следующий результат. это тест Многоуровневая непрямая адресация
Один указатель может указывать на другой, а тот, свою очередь, — на целевое зна чение. Это так называемая многоуровневая непрямая адресация, или применение указа телей на указатели. Такое применение указателей может показаться, на первый взгляд, запутанным. Для прояснения принципа многоуровневой непрямой адресации обра тимся за помощью к рис. 20.1. Как видите, значением обычного указателя является адрес переменной, содержащей требуемое значение. Если же применяется указатель на указатель, то первый из них содержит адрес второго, указывающего на переменную, содержащую требуемое значение.
Рис. 20.1. Одно- и многоуровневая непрямая адресация
Многоуровневая непрямая адресация может быть продолжена до любого предела, но потребность более чем в двух уровнях адресации по указателям возникает крайне редко. На самом деле чрезмерная непрямая адресация очень трудно прослеживается и чревата ошибками.
Переменная, являющаяся указателем на указатель, должна быть объявлена как та ковая. Для этого достаточно указать дополнительный знак * после имени типа пере менной. Например, в следующем объявлении компилятор уведомляется о том, что переменная q является указателем на указатель и относится к типу int. int** q;
Следует, однако, иметь в виду, что переменная q является указателем не на целое значение, а на указатель типа int.
Для доступа к целевому значению, косвенно адресуемому по указателю на указа тель, следует дважды применить оператор *, как в приведенном ниже примере. using System; class MultipleIndirect { unsafe static void Main { int x; // содержит значение типа int int* p; // содержит указатель типа int int** q; // содержит указатель на указатель типа int х = 10; р = &х; // поместить адрес переменной х в переменной р q = &р; // поместить адрес переменной р в переменной q Console.WriteLine(**q); // вывести значение переменной х } }
Результатом выполнения этой программы будет выведенное на экран значение 10 переменной х. В данной программе переменная р объявляется как указатель на значе ние типа int, а переменная q — как указатель на указатель типа int.
И последнее замечание: не путайте многоуровневую непрямую адресацию со структурами данных высокого уровня, в том числе связными списками, так как это со вершенно разные понятия. Массивы указателей
Указатели могут быть организованы в массивы, как и любой другой тип данных. Ниже приведен пример объявления массива указателей типа int длиной в три эле мента. int * [] ptrs = new int * [3];
Для того чтобы присвоить адрес переменной var типа int третьему элементу мас сива указателей, достаточно написать следующую строку кода. ptrs[2] = Svar;
А для того чтобы обнаружить значение переменной var, достаточно написать при веденную ниже строку кода. *ptrs[2] Оператор sizeof
Во время работы с небезопасным кодом иногда полезно знать размер в байтах одного из встроенных в C# типов значений. Для получения этой информации служит оператор sizeof. Ниже приведена его общая форма: sizeof(тип)
где тип обозначает тот тип, размер которого требуется получить. Вообще говоря, опе ратор sizeof предназначен главным образом для особых случаев и, в частности, для работы со смешанным кодом: управляемым и неуправляемым. Оператор stackalloc
Для распределения памяти, выделенной под стек, служит оператор stackalloc. Им можно пользоваться лишь при инициализации локальных переменных. Ниже приведена общая форма этого оператора: тип *р = stackalloc тип[размер]