Язык С не поддерживает отдельный строковый тип данных, но он позволяет определить строки двумя различными способами. В первом используется массив символов, а во втором – указатель на первый символ массива. Объявление char a[10]; указывает компилятору на необходимость резервирования места для максимум 10 символов. Константа а содержит адрес ячейки памяти, в который помещено значение первого из десяти объектов типа char. Процедуры, связанные с занесением конкретной строки в массив а, копируют ее по одному символу в область памяти, на которую указывает константа а, до тех пор, пока не будет скопирован нулевой символ, оканчивающий строку. Когда выполняется функция типа printf("%s",a);, ей передается значение а, т.е. адрес первого символа, на который указывает а. Если первый символ нулевой, то работа функции printf заканчивается, а если нет, то она выводит его на экран, прибавляет к адресу единицу и снова начинает проверку на нулевой символ. Такая обработка позволяет снять ограничения на длину строки (конечно, в пределах объявленной размерности): строка может быть любой длины до тех пор, пока есть место в памяти, куда ее можно поместить.
Вторым способом определения строки является использование указателя на символ. Объявление char *b; задает переменную b, которая может содержать адрес некоторого объекта. Однако в данном случае компилятор не резервирует место для хранения символов и не инициализирует переменную b конкретным значением. Когда компилятор встречает инструкцию вида b=“Москва”, он производит следующие действия. Во-первых, как и в предыдущем случае, создает в каком-либо месте объектного модуля строку Москва, за которой следует нулевой символ. Во-вторых, присваивает значение начального адреса этой строки (адрес символа М) переменной b. Функция printf(“%s”,b); работает так же, как и в предыдущем случае, осуществляя вывод символов до тех пор, пока не встретится заключительный нуль.
Массив указателей можно инициализировать, т.е. назначать его элементам конкретные адреса некоторых заданных строк при объявлении. Если объявить char my_char [3,7], в памяти выделится следущий блок памяти (рис.4.4).
Г
| Р
| И
| Ш
| А
| !
| \0
|
Э
| Т
| О
| \0
| \0
| \0
| \0
|
Я
| \0
| \0
| \0
| \0
| \0
| \0
|
Рис.4.4. Вид выделенного блока памяти при статическом выделении памяти
Если же объявить char * my-char [3] и инициализировать этими словами, то в памяти будет выделено, что значительно экономит память (рис.4.5).
Г
| Р
| И
| Ш
| А
| !
| \0
|
Э
| Т
| О
| \0
| |
Я
| \0
| |
Рис.4.5.Вид выделенного блока памяти при использовании указателя
Пример:
#include <stdio.h>
#include <string.h>
#include <conio.h>