Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
SAOD_Konspekt-I.doc
Скачиваний:
14
Добавлен:
03.11.2018
Размер:
2.32 Mб
Скачать

3.5. Строки

Строки как структуры данным могут быть организованы несколькими различными способами. Первый способ – дескрипторный. В этом случае строка представляет собой массив символов, первый элемент массива является дескриптором, т.е. содержит информацию о длине строки. Непосредственно символы строки хранятся, начиная только со второго элемента массива. Недостатком такого способа организации является ограниченная длина строки. Так, если каждый элемент является байтом, то длина строки ограничена 255 символами. Именно такой способ используется в Паскале для строк типа String.

Второй способ организации – маркерный. И в этом случае строка также представляется массивом символов, но символы хранятся, начиная с первого (начального) элемента массива, а заканчивается строка специальным символом, который называется маркером. Такой способ используется в Паскале для строк типа Askiiz и в Си/Си++. Длина строки при маркерном способе организации, теоретически, не ограниченна. Пример маркерного способа организации строк в Си/Си++ уже был приведён:

char vls2[] = "2nd very long string.\n";

Третий способ организации – в виде линейного связного списка. Такой способ будет рассмотрен позже.

В языках Си/Си++ указатели на тип char могут инициализироваться с помощью строковых констант:

char *vls1 = ”1st very long string.”;

Следует отметить, что присваивание для строковых констант (и вообще строк) возможно только при инициализации. После определения символьных массивов или указателей на тип char в программе для назначения им значений следует пользоваться библиотечной функцией strcpy:

char* str;

strcpy(str,”строка”);

Строки при этом обладают всеми свойствами массивов, а указатели – это обычные «указатели на массивы», значением такого указателя является адрес начального символа строки.

3.6. Массивы указателей

Существует возможность создать массив указателей. Определение вида

int* arp[20];

создает массив указателей на объекты типа int. Элементами этого массива являются указатели на объекты типа int.

Массивы указателей можно инициализировать, например массив строк:

char* month1[ ] = {”январь”,”февраль”,”март”};

Элементами массива month1 являются адреса начальных символов строк. Возможен 2-й вариант размещения данных (в том числе строк) в памяти, при котором расход памяти больше:

char month2[ ][10] = {”январь”,”февраль”,”март”};

Одной из областей применения массивов указателей является сортировка сложных объектов неодинакового размера.

3.7. Интерпретация составных описателей

Правило интерпретации составных описателей может быть названо чтением «изнутри – наружу». Начать интерпретацию нужно с идентификатора и проверить, есть ли справа от него открывающие скобки, квадратные [ или круглые (. Если да – то рассматривается правая часть описателя. Затем следует проверить, есть ли слева от идентификатора символ *, если да – то рассматривается левая часть. Если на каком либо шаге интерпретации справа встретится закрывающая круглая скобка ), которая используется для изменения порядка интерпретации, то необходимо сначала закончить интерпретацию внутри данной пары круглых скобок ( ), а затем продолжит интерпретацию справа от закрывающей круглой скобки. На последнем шаге интерпретируется спецификация типа. После этого тип объявленного объекта полностью известен.

char *(*(*var)())[10];

7 6 4 2 1 3 5

1. Идентификатор var – это

2. Указатель на

3. Функцию без аргументов, возвращающую

4. Указатель на

5. Массив из 10 элементов, являющихся

6. Указателями на

7. Значения типа char.

Рассмотрим примеры:

1. int *var[5]; // var – массив указателей на значения типа int;

2. int (*var)[5]; // var – указатель на массив значений типа int;

3. long *var(); // var – функция, возвращающая указатель на значения типа long;

4. long (*var)(); // var – указатель на функцию, возвращающую значение типа long;

5. struct both {

int a;

char b; } (*var[5])(); // var – массив указателей на функции, возвращающие структуры типа both.

Чаще определения структурных типов и их использование в программах оказывается разнесенным по исходному тексту. В таком случае этот пример будет выглядеть следующим образом (на языке C++):

struct both {

int a;

char b; };

both (*var[5])();

6. double (*var())[3]; // var – функция, возвращающая указатель на массив из 3-х значений типа double;

7. union sign {

int x;

char y; } **var[5][5]; // var – массив из 5-ти элементов, являющихся массивами указателей на указатели на объединения типа sign (двумерный массив указателей на указатели).

Ситуация, аналогичная примеру 5. С учетом раздельного определения объединения и переменной var, последнее будет выглядеть следующим образом:

sign **var[5][5];

8. sign *(*var[5])[5]; // var – массив из 5-ти элементов, являющихся указателями на массив указателей на объединения типа sign.

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