Массивы

Автор работы: Пользователь скрыл имя, 16 Мая 2013 в 11:07, курсовая работа

Описание

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

Содержание

1.Введение
1)История создания языка Pascal
2)Краткий обзор языка
2.Основная часть
1)Массив.
2)Одномерные массивы
3)Алгоритмы сортировки одномерных массивов
4)Массивы в языках Pascal и Basic
5)Массив в Бейсике
6)Массив в Паскале
7)Двумерные массивы Паскаля – матрицы
8)Описание двумерного массива Паскаля
9)Основные действия с двумерными массивами Паскаля
10)Ввод двумерного массива Паскаля
11)Вывод двумерного массива Паскаля на экран
12)Представление двумерного массива Паскаля в памяти
13)Методы доступа к элементам массивов
14)Индексный массив
15)Специфические типы массивов
16)Динамические массивы
17)Гетерогенные массивы
3.Заключение
1)Основная
2) Дополнительная
Список литературы
1)основная литература
2)дополнительная дополнительная

Работа состоит из  1 файл

Курсовая массивы.docx

— 56.80 Кб (Скачать документ)

procedure primer (a: matrix);

..............................

Вернемся теперь к нашим  процедурам.

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

Тогда заголовок нашей  процедуры будет выглядеть так:

Procedure vvod ( var m : matrix );

Для реализации вложенных  циклов в процедуре нам потребуются  локальные переменные-счетчики, например, k и h . Алгоритм заполнения матрицы уже обсуждался, поэтому не будем его повторять.

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

Procedure print ( m : matrix );

И вновь для реализации вложенных циклов внутри процедуры  нам потребуются счетчики, пусть  они называются так же – k и h . Алгоритм вывода матрицы на экран был описан выше, воспользуемся этим описанием.

Пример программы двумерного массива Паскаля

Program proizvedenie;

Type

Matrix=array [1..10, 1..10] of integer;

Var

A: matrix;

N, m, i, j: byte;

P: integer;

Procedure vvod (var m: matrix);

Var k , h : byte ;

Begin

 For i :=1 to n do {переменная n для процедуры является глобальной, а значит «известной»}

 For j :=1 to m do {переменная m для процедуры является глобальной, а значит «известной»}

 M[i,j]:= random(10);

End;

Procedure print (m: matrix);

Var k, h: byte;

Begin

For i:=1 to n do

begin

For j:=1 to m do

Write (M[i, j]: 4);

 Writeln;

 end ;

End ;

Begin {начало основной программы}

 Writeln (‘Введите размерность матрицы:’);

 Readln(N, M);

 Vvod(a);

14

Print(a);

P:=1;

For i:=1 to N do

For j:=1 to M do

If a[i, j]<>0 then p:=p*a[i, j];

 Writeln ( p );

End .

Методы доступа к элементам  массивов

В языке СИ между указателями  и массивами существует тесная связь. Например, когда объявляется массив в виде int array[25], то этим определяется не только выделение памяти для двадцати пяти элементов массива, но и для указателя с именем array, значение которого равно адресу первого по счету (нулевого) элемента массива, т.е. сам массив остается безымянным, а доступ к элементам массива осуществляется через указатель с именем array. С точки зрения синтаксиса языка указатель arrey является константой, значение которой можно использовать в выражениях, но изменить это значение нельзя.

Поскольку имя массива  является указателем допустимо, например, такое присваивание:

 int array[25];

 int *ptr;

 ptr=array;

Здесь указатель ptr устанавливается на адрес первого элемента масcива, причем присваивание ptr=arrey можно записать в эквивалентной форме ptr=&arrey[0].

Для доступа к элементам  массива существует два различных  способа. Первый способ связан с использованием обычных индексных выражений  в квадратных скобках, например, array[16]=3 или array[i+2]=7. При таком способе доступа записываются два выражения, причем второе выражение заключается в квадратные скобки. Одно из этих выражений должно быть указателем, а второе - выражением целого типа. Последовательность записи этих выражений может быть любой, но в квадратных скобках записывается выражение следующее вторым. Поэтому записи array[16] и 16[array] будут эквивалентными и обозначают элемент массива с номером шестнадцать. Указатель используемый в индексном выражении не обязательно должен быть константой, указывающей на какой-либо массив, это может быть и переменная. В частности после выполнения присваивания ptr=array доступ к шестнадцатому элементу массива можно получить с помощью указателя ptr в форме ptr[16] или 16[ptr].

Второй способ доступа  к элементам массива связан с  использованием адресных выражений  и операции разадресации в форме *(array+16)=3 или *(array+i+2)=7. При таком способе доступа адресное выражение равное адресу шестнадцатого элемента массива тоже может быть записано разными способами *(array+16) или *(16+array).

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

15

Для приведенных примеров array[16] и 16[array] преобразуются в *(array+16).

Для доступа к начальному элементу массива (т.е. к элементу с  нулевым индексом) можно использовать просто значение указателя array или ptr. Любое из присваиваний

*array = 2;

 array[0] = 2;

*(array+0) = 2;

*ptr = 2;

 ptr[0] = 2;

*(ptr+0) = 2;

присваивает начальному элементу массива значение 2, но быстрее всего  выполнятся присваивания *array=2 и *ptr=2, так  как в них не требуется выполнять  операции сложения.

Указатели на многомерные  массивы

Указатели на многомерные  массивы в языке СИ - это массивы  массивов, т.е. такие массивы, элементами которых являются массивы. При объявлении таких массивов в памяти компьютера создается несколько различных  объектов. Например при выполнении объявления двумерного массива int arr2[4][3] в памяти выделяется участок для хранения значения переменной arr, которая является указателем на массив из четырех указателей. Для этого массива из четырех указателей тоже выделяется память. Каждый из этих четырех указателей содержит адрес массива из трех элементов типа int, и, следовательно, в памяти компьютера выделяется четыре участка для хранения четырех массивов чисел типа int, каждый из которых состоит из трех элементов. Такое выделение памяти показано на схеме на arr  

Распределение памяти для  двумерного массива.

arr[0]      а             arr[0][0] arr[0][1] arr[0][2]

arr[1]      а arr[1][0] arr[1][1] arr[1][2]

arr[2]      а             arr[2][0] arr[2][2] arr[2][1]

arr[3]      а arr[3][0] arr[3][1] arr[3][2]

Таким образом, объявление arr2[4][3] порождает в программе три  разных объекта: указатель с идентификатором  arr, безымянный массив из четырех указателей и безымянный массив из двенадцати чисел типа int. Для доступа к безымянным массивам используются адресные выражения с указателем arr. Доступ к элементам массива указателей осуществляется с указанием одного индексного выражения в форме arr2[2] или *(arr2+2). Для доступа к элементам двумерного массива чисел типа int должны быть использованы два индексных выражения в форме arr2[1][2] или эквивалентных ей *(*(arr2+1)+2) и (*(arr2+1))[2]. Следует учитывать, что с точки зрения синтаксиса языка СИ указатель arr и указатели arr[0], arr[1], arr[2], arr[3] являются константами и их значения нельзя изменять во время выполнения программы.

Размещение трехмерного  массива происходит аналогично и  объявление float arr3[3][4][5] порождает в программе кроме самого трехмерного массива из шестидесяти чисел типа float массив из четырех указателей на тип float, массив из трех указателей на массив указателей на float, и указатель на массив массивов указателей на float.

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

Например, обращение к  элементу arr2[1][2] можно осуществить  с помощью указателя ptr2, объявленного в форме int *ptr2=arr2[0] как обращение ptr2[1*4+2] (здесь 1 и 2 это индексы используемого элемента, а 4 это число элементов в строке) или как ptr2[6]. Заметим, что внешне похожее обращение arr2[6] выполнить невозможно так как указателя с индексом 6 не существует.

Для обращения к элементу arr3[2][3][4] из трехмерного массива тоже можнo использовать указатель, описанный как float *ptr3=arr3[0][0] с одним индексным выражением в форме ptr3[3*2+4*3+4] или ptr3[22].

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

 struct INDEX { int i,

 int j,

 int k } min_index ;

 struct INDEX * find_min (int *ptr1, int l, int m int n)

{ int min, i, j, k, ind;

min=*ptr1;

 min_index.i=min_index.j=min_index.k=0;

for (i=0; i*(ptr1+ind)

{ min=*(ptr1+ind);

 min_index.i=i;

 min_index.j=j;

 min_index.k=k;

 }

}

 return &min_index;

}

 Операции с указателями

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

Пример:

 int *ptr, a[10];

ptr=&a[5];

 ptr++; /* равно адресу элемента a[6] */

 ptr--; /* равно адресу элемента a[5] */

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

Пример:

 int *ptr1, *ptr2, a[10];

 int i=2;

ptr1=a+(i+4); /* равно адресу элемента a[6] */

 ptr2=ptr1-i; /* равно адресу элемента a[4] */

В операции вычитания могут  участвовать два указателя на один и тот же тип. Результат такой  операции имеет тип int и равен числу элементов исходного типа между уменьшаемым и вычитаемым, причем если первый адрес младше, то результат имеет отрицательное значение.

Пример:

 int *ptr1, *ptr2, a[10];

 int i;

ptr1=a+4;

ptr2=a+9;

 i=ptr1-ptr2; /* равно 5 */

 i=ptr2-ptr1; /* равно -5 */

Значения двух указателей на одинаковые типы можно сравнивать в операциях ==, !=, <, <=, >, >= при этом значения указателей рассматриваются просто как целые числа, а результат сравнения равен 0 (ложь) или 1 (истина).

Пример:

 int *ptr1, *ptr2, a[10];

ptr1=a+5;

ptr2=a+7;

if (prt1>ptr2) a[3]=4;

В данном примере значение ptr1 меньше значения ptr2 и поэтому  оператор a[3]=4 не будет выполнен.

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

В языке СИ элементы массивов могут иметь любой тип, и, в  частности, могут быть указателями  на любой тип. Рассмотрим несколько  примеров с использованием указателей.

Следующие объявления переменных

 int a[]={10,11,12,13,14,};

 int *p[]={a, a+1, a+2, a+2, a+3, a+4};

 int **pp=p;

порождают программные объекты, представленные на схеме

pp     pа    .        .        .        .        .

в в    в в в

18

aа      11 12 13    14 15

Схема размещения переменных при объявлении.

При выполнении операции pp-p получим нулевое значение, так как ссылки pp и p равны и указывают на начальный элемент массива указателей, связанного с указателем p ( на элемент p[0]).

После выполнения операции pp+=2 схема изменится и примет вид, изображенный                  

 pp pа .      .        .        .

 в в в в в

 aа     10 11 12 13 14

Схема размещения переменных после выполнения операции pp+=2.

Результатом выполнения вычитания  pp-p будет 2, так как значение pp есть адрес третьего элемента массива p. Ссылка *pp-a тоже дает значение 2, так как обращение *pp есть адрес третьего элемента массива a, а обращение a есть адрес начального элемента массива a. При обращении с помощью ссылки **pp получим 12 - это значение третьего элемента массива a. Ссылка *pp++ даст значение четвертого элемента массива p т.е. адрес четвертого элемента массива.

Если считать, что pp=p, то обращение *++pp это значение первого элемента массива a (т.е. значение 11), операция ++*pp изменит содержимое указателя p[0], таким образом, что он станет равным значению адреса элемента a[1].

Сложные обращения раскрываются изнутри. Например обращение *(++(*pp)) можно разбить на следующие действия: *pp дает значение начального элемента массива p[0], далее это значение инкременируется ++(*p) в результате чего указатель p[0] станет равен значению адреса элемента a[1], и последнее действие это выборка значения по полученному адресу, т.е. значение 11.

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

 int a[3][3]={ { 11,12,13 },

 { 21,22,23 },

{ 31,32,33 } };

 int *pa[3]={ a,a[1],a[2] };

 int *p=a[0];

Схема размещения указателей на двумерный массив.

Согласно этой схеме доступ к элементу a[0][0] получить по указателям a, p, pa при помощи следующих ссылок: a[0][0], *a, **a[0], *p, **pa, *p[0].

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

 char *c[]={ "abs", "d", "yes", "no" };

char **cp[]={ c+3, c+2 , c+1 , c };

 char ***cpp=cp;

Схема размещения указателей на строки.

 

Динамическое размещение массивов

При динамическом распределении  памяти для массивов следует описать  соответствующий указатель и  присваивать ему значение при  помощи функции calloc. Одномерный массив a[10] из элементов типа float можно создать следующим образом

 float *a;

a=(float*)(calloc(10,sizeof(float));

Для создания двумерного массива  вначале нужно распределить память для массива указателей на одномерные массивы, а затем распределять память для одномерных массивов. Пусть, например, требуется создать массив a[n][m], это можно сделать при помощи следующего фрагмента программы:

 #include

main ()

{ double **a;

int n,m,i;

scanf("%d %d",&n,&m);

a=(double **)calloc(m,sizeof(double *));

for (i=0; i<=m; i++)

a[i]=(double *)calloc(n,sizeof(double));

 . . . . . . . . . . . .

}

Аналогичным образом можно  распределить память и для трехмерного  массива размером n,m,l. Следует только помнить, что ненужную для дальнейшего выполнения программы память следует освобождать при помощи функции free.

 #include

main ()

{ long ***a;

int n,m,l,i,j;

scanf("%d %d %d",&n,&m,&l);

/* -------- распределение памяти -------- */

a=(long ***)calloc(m,sizeof(long **));

for (i=0; i<=m; i++)

{ a[i]=(long **)calloc(n,sizeof(long *));

for (j=0; i<=l; j++)

a[i][j]=(long *)calloc(l,sizeof(long));

}

. . . . . . . . . . . .

/* --------- освобождение памяти ----------*/

for (i=0; i<=m; i++)

{ for (j=0; j<=l; j++)

free (a[i][j]);

free (a[i]);

 }

 

free (a);

}

Рассмотрим еще один интересный пример, в котором память для массивов распределяется в вызываемой функции, а используется в вызывающей. В  таком случае в вызываемую функцию  требуется передавать указатели, которым  будут присвоены адреса выделяемой для массивов памяти.

 Пример:

#include

main()

{ int vvod(double ***, long **);

 double **a; /* указатель для массива a[n][m] */

 long *b; /* указатель для массива b[n] */

 vvod (&a,&b);

.. /* в функцию vvod передаются адреса указателей, */

 .. /* а не их значения */

..

}

int vvod(double ***a, long **b)

{ int n,m,i,j;

scanf (" %d %d ",&n,&m);

*a=(double **)calloc(n,sizeof(double *));

*b=(long *)calloc(n,sizeof(long));

for (i=0; i<=n; i++)

*a[i]=(double *)calloc(m,sizeof(double));

 .....

}

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

Пример:

#include

int main()

{ float *q, **b;

int i, j, k, n, m;

scanf("%d %d",&n,&m);

q=(float *)calloc(m,sizeof(float));

 /* сейчас указатель q показывает на начало массива */

 q[0]=22.3;

q-=5;

/* теперь начальный элемент  массива имеет индекс 5, */

/* а конечный элемент  индекс n-5 */

 q[5]=1.5;

/* сдвиг индекса не  приводит к перераспределению  */

Информация о работе Массивы