Инициализация массивов и классы памяти

Инициализация массивов и классы памяти

     Для хранения данных, необходимых программе, часто используют массивы. Например, в массиве из 12 элементов можно хранить информацию о количестве дней каждого месяца. В подобных случаях желательно иметь удобный способ инициализации массива перед началом работы программы. Такая возможность, вообще говоря, существует, но только для статической и внешней памяти. Давайте посмотрим, как она используется.

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

int fix = 1;

float flах = PI*2;

при этом предполагается, что PI - ранее введенное макроопределение. Можем ли мы делать что-либо подобное с массивом? Ответ не однозначен: и да, и нет.

     Внешние и статические массивы можно инициализировать.

     Автоматические и регистровые массивы инициализировать нельзя.

     Прежде чем попытаться инициализировать массив, давайте посмотрим, чтo там находится, если мы в него ничего не записали.

/* проверка содержимого массива */

main( ) {

int fuzzy[2]; /*автоматический массив */

static int wuzzy[2]; /* статический массив */

printf("%d %d ", fuzzy[1], wuzzy[1];

}

Программа напечатает

525 0

     Полученный результат иллюстрирует следующее правило:

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

     Прекрасно! Теперь мы знаем, что нужно предпринять для обнуления статического или внешнего массива - просто ничего не делать. Но как быть, если нам нужны некоторые значения, отличные от нуля, например количество дней в каждом месяце. В этом случае мы можем делать так:

/* дни месяца */

int days[12]=[31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];

main( )

{

int index;

extern int days[ ]; /* необязательное описание */

for(index = 0; index < 12; index++)

printf(" Месяц %d имеет %d дней. ", index + 1, days[index]);

}

Результат:

Месяц 1 имеет 31 дней. Месяц 2 имеет 28 дней. Месяц 3 имеет 31 дней.

Месяц 4 имеет 30 дней. Месяц 5 имеет 31 дней. Месяц 6 имеет 30 дней.

Месяц 7 имеет 31 дней. Месяц 8 имеет 31 дней. Месяц 9 имеет 30 дней.

Месяц 10  имеет 31 дней. Месяц 11 имеет 30 дней. Месяц 12 имеет 31 дней.

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

     Определив массив days[ ] вне тела функции, мы тем самым сделали его внешним. Мы инициировали его списком, заключенным в скобки, используя при этом запятые для разделения элементов списка.

     Количество элементов в списке должно соответствовать размеру массива. А что будет, если мы ошиблись в подсчете? Попробуйте переписать последний пример, используя список, который короче, чем нужно (на два элемента):

/* дни месяца */

int days[12]=[31, 28, 31, 30, 31, 30, 31, 31, 30, 31];

main( )

{

int index;

extern int days[  ]; /* необязательное описание */

for(index = 0; index < 12; index++)

printf(" Месяц %d имеет %d дней. ", index + 1, days[index]);

}

В этом случае результат оказывается иным:

Месяц 1 имеет 31 дней.

Месяц 2 имеет 28 дней.

Месяц 3 имеет 31 дней.

Месяц 4 имеет 30 дней.

Месяц 5 имеет 31 дней.

Месяц 6 имеет 30 дней.

Месяц 7 имеет 31 дней.

Месяц 8 имеет 31 дней.

Месяц 9 имеет 30 дней.

Месяц 10 имеет 31 дней.

Месяц 11 имеет 0 дней.

Месяц 12 имеет 0 дней.

     Можно видеть, что у компилятора не возникло никаких проблем: просто когда он исчерпал весь список с исходными данными, то стал присваивать всем остальным элементам массива нулевые значения.

     Однако в случае излишне большого списка компилятор будет уже не столь "великодушен" к вам, поскольку посчитает выявленную избыточность ошибкой. Поэтому нет никакой необходимости заранее подвергать себя "насмешкам" компилятора. Надо просто выделить массив, размер которого будет достаточен для размещения списка:

/* дни месяца */

int days[ ]=[31, 28, 31, 30, 31, 30, 31, 31, 30, 31];

main( )

{

int index;

extern int days[ ]; /* необязательное описание */

for(index = 0; index < sizeof days/(sizeof (int)); index++)

printf(" Месяц %d имеет %d дней. ", index + 1, days [index]);

}

К этой программе следует сделать два существенных замечания:

    Первое: если вы используете пустые скобки для инициализации массива, то компилятор сам определит количество элементов в списке и выделит для него массив нужного размера.

    Второе: оно касается добавления, сделанного п управляющем операторе for. He полагаясь (вполне обоснованно) на свои вычислительные способности, мы возложили задачу подсчета размера массива на компилятор. Оператор sizeof определяет размер в байтах объекта или типа, следующего за ним. (Мы. уже упоминали об этом в гл. 3.) В нашей вычислительной системе размер каждого элемента типа int равен двум байтам, поэтому для получения количества элементов массива мы делим общее число байтов, занимаемое массивом, на 2. Однако в других системах элемент типа int может иметь иной размер. Поэтому в общем случае выполняется деление на значение переменной sizeof (для элемента типа int). Ниже приведены результаты работы нашей программы:

Месяц 1 имеет 31 дней.

Месяц 2 имеет 28 дней.

Месяц 3 имеет 31 дней.

Месяц 4 имеет 30 дней.

Месяц 5 имеет 31 дней.

Месяц 6 имеет 30 дней.

Месяц 7 имеет 31 дней.

Месяц 8 имеет 31 дней.

Месяц 9 имеет 30 дней.

Месяц 10 имеет 31 дней.

 Ну вот, теперь мы получаем точно 10 значений. Наш метод, позволяющий программе самой находить размер массива, не позволил нам напечатать конец массива.

     Существует и более короткий способ инициализации массивов, но поскольку он применим только к символьным строкам, мы рассмотрим его в следующей главе.

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

/* присваивание значений массиву */

main( )

{

int counter, evens [50];

for(counter = 0; counter < 50; counter++)

evens[counter] = 2 * counter;

...

}

Поделитесь на страничке

Следующая глава >

Похожие главы из других книг

4.4 Классы Памяти

Из книги C++ автора Хилл Мюррей

4.4 Классы Памяти Есть два описываемых класса памяти: автоматический и статический.Автоматические объекты локальны для каждого вызова блока и сбрасываются по выходе из него.Статические объекты существуют и сохраняют свое значение в течение выполнения всей


Создание массивов

Из книги AutoCAD 2009 автора Орлов Андрей Александрович

Создание массивов Массивом называют множество однородных предметов, образующих единое целое. Массивы программы AutoCAD – это совокупность копий одного объекта, расположенных на равном расстоянии друг от друга. Так как массивы связаны со смещением координат, они могут быть


R.3.5 Классы памяти

Из книги Справочное руководство по C++ автора Страустрап Бьярн

R.3.5 Классы памяти Существует два описываемых класса памяти: автоматический и статический.Автоматические объекты становятся локальными при передаче управления в каждый блок.Статические объекты существуют и сохраняют свое значение во все время выполнения


Создание массивов

Из книги AutoCAD 2010 автора Орлов Андрей Александрович

Создание массивов Массивом называют множество однородных предметов, образующих единое целое. Массивы программы AutoCAD – это совокупность копий одного объекта, расположенных на равном расстоянии друг от друга. Так как массивы связаны со смещением координат, они могут быть


Ревизия массивов

Из книги Основы объектно-ориентированного программирования автора Мейер Бертран

Ревизия массивов Набросок библиотечного класса ARRAY дан в предыдущей лекции. Теперь мы в состоянии дать ему подходящее определение. Фундаментальное понятие массива требует задания предусловий, постусловий и инварианта.Приведем улучшенный, но все еще схематичный


Отложенные классы как частичные интерпретации: классы поведения

Из книги Программирование на языке Ruby [Идеология языка, теория и практика применения] автора Фултон Хэл

Отложенные классы как частичные интерпретации: классы поведения Не все отложенные классы так близки к АТД как STACK. В промежутке между полностью абстрактным классом, таким как STACK, в котором все существенные компоненты отложены, и эффективным классом, таким как FIXED_STACK,


8.1.4. Сравнение массивов

Из книги Справочник по PHP автора

8.1.4. Сравнение массивов При сравнении массивов возможны неожиданности — будьте осторожны!Для сравнения массивов служит метод экземпляра <=>. Он работает так же, как в других контекстах, то есть возвращает -1 (меньше), 0 (равно) или 1 (больше). Методы == и != опираются на


8.1.22. Чередование массивов

Из книги VBA для чайников автора Каммингс Стив

8.1.22. Чередование массивов Предположим, что есть два массива и надо построить из них третий, который содержит массивы из двух элементов, взятых из соответственных позиций исходных массивов. В последних версиях Ruby модуль Enumerable содержит метод zip:a = [1, 2, 3, 4]b = ["a", "b", "c", "d"]с =


Сортировка массивов

Из книги Язык программирования Си для персонального компьютера автора Бочков C. О.

Сортировка массивов array_reverseРасстановка элементов массива в обратном порядке.Синтаксис:array array_reverse(array arr [, bool preserve_keys])Функция array_reverse() возвращает массив, элементы которого следуют в обратном порядке относительно массива, переданного в параметре. При этом связи между


Классы памяти

Из книги автора

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


10. Классы памяти и разработка программ

Из книги автора

10. Классы памяти и разработка программ ЛОКАЛЬНЫЕ И ГЛОБАЛЬНЫЕ ПЕРЕМЕННЫЕКЛАССЫ ПАМЯТИФУНКЦИЯ ПОЛУЧЕНИЯ СЛУЧАЙНЫХ ЧИСЕЛПРОВЕРКА ОШИБОКМОДУЛЬНОЕ


Резюме: Классы памяти

Из книги автора

Резюме: Классы памяти I. Ключевые слова: auto, extern, static, registerII. Общие замечания:     Класс памяти определяет область действия переменной и продолжительность ее существования в памяти. Класс памяти устанавливается при описании переменной с соответствующим ключевым словом.


УКАЗАТЕЛИ МАССИВОВ

Из книги автора

УКАЗАТЕЛИ МАССИВОВ      Как было сказано в гл. 9, указатели позволяют нам работать с символическими адресами. Поскольку в реализуемых аппаратно командах вычислительной машины интенсивно используются адреса, указатели предоставляют возможность применять адреса


ПРИЛОЖЕНИЕ Г. ТИПЫ ДАННЫХ И КЛАССЫ ПАМЯТИ  

Из книги автора

ПРИЛОЖЕНИЕ Г. ТИПЫ ДАННЫХ И КЛАССЫ ПАМЯТИ   Основные типы данных Ключевые слова: Основные типы данных определяются с помощью следующих семи ключевых слов: int, long, short, unsigned, char, float, double Целые со знаком: Могут иметь положительные и отрицательные значения.int: основной тип


Классы памяти

Из книги автора

Классы памяти I.Ключевые слова: auto, external, static, registerII. Основные замечания Класс памяти переменной определяет область ее действия и продолжительность использования. Класс памяти определяется местом задания переменной и соответствующим ключевым словом. Переменные,