Студопедия

Главная страница Случайная страница

КАТЕГОРИИ:

АвтомобилиАстрономияБиологияГеографияДом и садДругие языкиДругоеИнформатикаИсторияКультураЛитератураЛогикаМатематикаМедицинаМеталлургияМеханикаОбразованиеОхрана трудаПедагогикаПолитикаПравоПсихологияРелигияРиторикаСоциологияСпортСтроительствоТехнологияТуризмФизикаФилософияФинансыХимияЧерчениеЭкологияЭкономикаЭлектроника






Вопрос № 66. Динамическое размещение данных в программировании.






Память в работающей программе с точки зрения времени ее выделения может распределяться:

1) статически (секция данных, стэк) – память распределяется в момент компиляции. (пример: int x[10];)

2)динамически – для дин.размещения памяти выделяется спец. Область памяти КУЧА. Объем динамически размещаемых данных опр-ся лишь в момент выполнения программы. Для работы с дин.памятью исп-ся спец.библиотечные ф-ии, расположенные в заголовочном файле #include < alloc.h>.

 

В Си существуют 2 способа доступа к переменной: ссылка на переменную по имени и использование механизма указателей.

Ук.-переменная-это переменная, предназначенная для хранения адреса в памяти.

Ук.-константа – это значение адреса оперативной памяти.

Синтаксис:

[< имя типа> ] * < имя переменной>

*-операция косвенной адресации - операция разыменования указателя - Это операция обращения к содержимому памяти по адресу, к-ый хранится в переменной-указателе. Операнд – указатель. Результат – тот объект, который адресует указатель – операнд.

Значением выражения *р будет число, которое хранится по адресу р.

& - операция взятия адреса применима только к объектам, размещенным в памяти и имеющим имя. Ее нельзя применить к выражениям, константам. Значение – адрес того объекта, к которому операция применяется.

Int date = 1974; //Объявлена пер-я с инициализацией

Int *p = & date; // Объявлен указатель на тип int, к-ый инициализирован адресом пер-ой date

= - операция присваивания. Можно присваивать только значения указателей одинаковых типов.

Int i, *p1, *p2=& i;

Float s, *p3;

P1=p2; //Ук.р1 получил значение указателя р2, т.е.стал равен адресу переменной i

P3=p1; //нельзя

Операции адресной арифметики. К ук.применимы орерации суммирования, вычитания, инкремента и декремента. Имеется ограничение на использование операции суммирования: нельзя складывать две переменные-указатели. Можно прибавить к указателю целую величину.(р+1). При этом важной особенностью адресной арифметики является то, что вычисляемое значение зависит от типа указателя, т.е.от размера того объекта, на который указатель ссылается. Пример:

Long b, *p3, *p4=& b; p3=p4+2; //если р4=0А01, то р3=0А09, если длина типа – 4 байта.

Операцию вычитание можно применять не только к ук.и целой величине, но к двум указателям на объекты одного типа. Разность между ук.также зависит оттипа ук. Значение разности м/у ук.имеет тип int. Это значение вычисляется в единицах, кратных длине отдельного элемента данных того типа, к которому отнесен указатель. Пример:

Int x[5]; int *I, k, j; i= & x[o]; k= & x[5]; j=k-I; // j =4.

Разрешается сравнивать указатели одного типа.

Связь массивов с указателями.

Имя массива является ук.-констаной, равной адресу начала массива. Это позволяет осуществить другой способ доступа к элементам массива – с помощью механизма указателей. Используя операцию косвенной адресации (*), можно получить доступ любому элементу массива.

Int a[5]; a[2]~*(a+2) a тождественно & a[0]

a[i]~a(a+i) a+i тождественно & a[i]

Указатель это пер-я, которая сама хранится в памяти, как и любая другая переменная, занимая либо 2, либо 4 байта.-> указатель на указатель: int **p;

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

Имя двумерного массива является указателем-констант на массив указателей-констант. Элементами такого массива являются указатели-константы на начало каждой иэ строк двумерного массива. Пример: int [3] [3] указателями-константами на нулевую, 1, 2 строки будут m[0], m[1], m[2], следующие выражения будут тождественными:

m[0] и & m[0] [0]

m[2] и & m[2] [0]

m и & m[0]

m+2 и & m[2]

Под динамической памятью понимается память, отличная от памяти, выделенной для кода функции, статических данных и не находящуюся в данный момент под управлением средств организации стека программы. Управление дин-ой памятью полностью берет на себя программист. Почти серьезная программа использует динамическую память. Это объясняется двумя обстоятельствами: во-первых, только при использовании динамической памяти возможно экономное и достаточно эффективное распределение ограниченных ресурсов памяти ЭВМ, во-вторых, поведение дин-их объектов полностью определяется программистом.

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

Void* malloc (unsigned s); - выделяет память. Функция возвращает указатель на начало области дин-ой памяти длиной в s байт. В куче выделяется запрошенное число байт. Для универсальности тип возвращаемого функцией значения есть void*. Указатель такого типа перед использованием надо преобразовать к указателю другого типа спомощью операции явного приведения.

Int * x; x=(int*)malloc(1024);

Void* calloc (unsigned n, unsigned m); - выделяет память и инициализирует ее нулевыми значениями. Ф-я возвращает ук.на начало области обнуленной дин-ой памяти для размещения n элементов по m байт каждый.

Void* realloc (void* bl, unsigned new-bl); - распределяет память. Ф-я изменяет размер блока ранее выделенной дин-ой памяти до размера new-bl байт. Bl – адрес начала изменяемого блока.

Void free (void* bl); - освобождает ранее выделенный участок памяти. Bl – первого байта освобождаемого блока памяти. Ф-я не имеет возвращаемого значения.



Поделиться с друзьями:

mylektsii.su - Мои Лекции - 2015-2025 год. (0.007 сек.)Все материалы представленные на сайте исключительно с целью ознакомления читателями и не преследуют коммерческих целей или нарушение авторских прав Пожаловаться на материал