- •Министерство образования Республики Беларусь
- •1. Что такое программа на языке программирования
- •2. Общее знакомство с языком с
- •3. Структура простой программы на языке с
- •4. Что такое программа на языке с
- •5. Представление информации и типы данных в языке с
- •6. Константы
- •7. Переменные
- •8. Элементарный вВод и вывод информации
- •9. Выражения и операции
- •9.1. Арифметические операции
- •9.2. Операция изменения знака
- •9.3. Операции инкремента и декремента
- •9.4. Операция присваивания
- •9.6. Поразрядные логические операции
- •9.8. Логические операции и операции отношения
- •9.9. Условная операция «? :»
- •9.10. Операция последовательного вычисления
- •9.11. Операция определения требуемой памяти в байтах sizeof
- •9.12. Операция приведения типа (type)
- •10. Операторы управления вычислительным процессом
- •10.1. Операторы ветвления if и else
- •10.2. Оператор switch
- •10.3. Оператор цикла while
- •10.4. Оператор цикла do…while
- •10.5. Оператор цикла for
- •10.6. Бесконечные циклы
- •10.7. Другие управляющие средства языка с
- •10.8. Стандартные математические функции
- •11. Вычисление выражений и побочные эффекты
- •11.1. Преобразования типов при вычислении выражений
- •11.2. Побочные эффекты при вычислении выражений
- •12. Массивы
- •12.1. Описание массива
- •12.2. Инициализация массива
- •12.3. Ввод-вывод массива
- •12.4. Двумерные массивы (массивы массивов)
- •13. Указатели
- •14. Адресная арифметика
- •15. Массивы и указатели
- •15.1. Указатели и одномерные массивы
- •15.2. Указатели и двумерные массивы
- •16. Строки
- •17. Массивы строк
- •18. Функции
- •18.1. Определение функции в языке с
- •18.2. Возвращение значений из функции
- •18.3. Формальные и фактические параметры функции
- •18.4. Вызов функции
- •18.5. Объявление и определение функции: прототип функции
- •19. Передача параметров в функции
- •19.1. Способы передачи параметров в функции
- •19.2. Передача параметров в функции в языке с
- •19.3. Передача указателей в функции
- •20. Классы хранения и видимость переменных
- •20.1. Общие положения
- •20.2. Спецификаторы класса памяти
- •20.3. Область видимости функций
- •20.4. Глобальные переменные
- •20.5. Глобальные статические переменные
- •20.6. Локальные переменные
- •20.7. Статические локальные переменные
- •20.8. Регистровые переменные
- •20.9. Выводы
- •21. Организация памяти программы
- •22. Многофайловая компиляция (проекты)
- •23. Передача в функции массивОв
- •23.1. Передача одномерных массивов в функции
- •23.2. Передача двумерных массивов в функции
- •23.3. Передача в функции символьных строк
- •23.4. Возвращение указателей из функций
- •24. Функции с переменным количеством аргументов
- •24.1. Соглашения о вызовах: модификаторы функций
- •24.2. Объявление списка параметров переменной длины
- •25. Передача параметров в функцию main()
- •26. Указатели на функцию
- •27. Стандартные функцИи языка с
- •27.1. Функции для работы со строками
- •27.2. Функции для проверки символов и преобразования данных
- •27.3. Функция быстрой сортировки – gsort()
- •27.4. Функция двоичного поиска – bsearch()
- •28. Работа с файлами
- •28.1. Основные понятия
- •28.2. Основные функции для работы с файлами
- •28.3. Открытие и закрытие файлов
- •28.4. Ввод/вывод символов
- •28.5. Ввод/вывод строк
- •28.6. Форматированный ввод/вывод
- •28.7. Ввод/вывод блоков данных
- •28.8. Другие средства для работы с файлами
- •28.9. Ввод/вывод низкого уровня (префиксный доступ к файлам)
- •29. Типы, определяемые пользователем: Перечисления, структуры и объединения
- •29.1. Переименование типов – оператор typedef
- •29.2. Перечисления (enum)
- •29.3. Основные сведения о структурах
- •29.4. Структурные переменные в памяти компьютера
- •29.5. Доступ к полям структуры
- •29.6. Массивы структур
- •29.7. Структуры и функции
- •29.8. Объединения (union)
- •30. Динамическая память
- •30.1. Понятие динамического объекта
- •30.2 Создание и уничтожение динамических объектов
- •30.3 Динамическое размещение одномерных массивов и строк
- •30.4 Динамическое размещение двумерных массивов
- •30.5. Функции для работы с блоками памяти
- •31. Динамические структуры данных
- •31.1. Понятие структуры данных
- •31.2. Структуры, ссылающиеся на себя
- •31.3. Связанные списки
- •31.5. Очереди
- •Ниже приводятся примеры функций для очереди (структура элемента очереди совпадает со структурой элемента стека в примере выше):
- •32. Препроцессор языка с
- •32.1 Директива включения файлов
- •32.2. Директива определения макрокоманд (макросов)
- •32.3 Директива условной компиляции
- •32.4 Дополнительные директивы препроцессора
30.3 Динамическое размещение одномерных массивов и строк
Функции для работы с динамической памятью позволяют использовать массивы с границами, задаваемыми переменными, а не константами.
При динамическом распределении памяти для одномерных массивов следует описать указатель соответствующего типа и присвоить ему значение при помощи фунций malloc() или calloc().
Одномерный массив из n целых чисел можно создать двумя способами:
int *a, *b;
int n = 10;
a = (int *)malloc(n * sizeof(int));
b = (int *)calloc(n, sizeof(int)); // инициализация 0
Освобождение памяти, выделенной под массивы:
free(a);
free(b);
Обратить внимание! Адреса освобождаемых массивов должны строго совпадать с адресами, которые вернули функции динамического размещения массивов.
int *a, n = 10;
a = (int *)malloc(n * sizeof(int));
a += 2; // перешли к элементу массива с индексом 2
*a = 100; // a[2] = 100
free(a); // ошибка, т.к. а != адрес начала массива
int *a, *p, n = 10;
a = (int *)malloc(n * sizeof(int));
p = a; // сохранение адреса начала массива
a += 2;
*a = 100;
free(p); // ошибки нет
Задача. Пример программы для работы с динамическим одномерным массивом целых чисел.
// выделение памяти и ввод массива
int vvod(int **m) {
int nn;
scanf ("%d", &nn);
*m = (int *)calloc(nn, sizeof(int));
// ввод массива
for (int i=0; i<nn; i++)
scanf("%d", *m+i); // *m+i == &(*m)[i]
return nn;
}
// поиск минимального элемента в массиве
int fmin(int *m, int nn) {
int min = 0x7FFF;
for (int i=0; i<nn; i++)
if (m[i] < min)
min = *(m+i);
return min;
}
void main() {
int n, i, min;
int *b; // указатель для массива b[n]
n = vvod(&b); // передается адрес указателя, а не его значение
// печать массива
for (i=0; i<n; i++)
printf("%d ", b[i]); // b[i] == *(b+i)
min = fmin(b, n);
printf("\nmin = %d\n", min);
free(b); // освобождение памяти
}
Задача. Пример программы для работы с динамическим одномерным массивом структур: создание массива, ввод, вывод на печать, поиск книги с минимальной стоимостью, освобождение памяти.
void main() {
struct Book {
char name[20];
int cost;
};
struct Book *p, min;
p = (struct Book *)malloc(5 * sizeof(struct Book));
for (int i=0; i<5; i++)
scanf("%s %d", (p+i)->name, &(p+i)->cost);
for (i=0; i<5; i++)
printf("%s %d\n", p[i].name, p[i].cost);
min = p[0];
for (i=1; i<5; i++)
if (min.cost > (p+i)->cost)
min = *(p+i);
printf("min: %s %d\n", min.name, min.cost);
free(p);
}
Выделение памяти под динамические строки ничем не отличается от выделения памяти под одномерные массивы других типов, только не следует при этом забывать про ноль-символ.
Задача. Пример программы для работы с динамической строкой.
char *strsave(char *s) {
char *p = NULL;
p = (char *)malloc(strlen(s) + 1);
if (p != NULL) strcpy(p, s);
return p;
}
void main() {
char s1[80], *s2;
gets(s1);
s2 = strsave(s1);
if (s2 != NULL) {
puts(s2);
free(s2);
}
}
30.4 Динамическое размещение двумерных массивов
При работе с динамическими двумерными массивами возникают определенные трудности, связанные с тем, что в языке С нет встроенных средств для учета длины строки двумерного массива при индексации. Поэтому программист сам должен обеспечить возможность индексации двумерного массива.
Есть два стандартных подхода к динамическому созданию двумерного массива: работа с двумерным массивом как с одномерным и имитация устройства двумерного массива как массива массивов. Рассмотрим оба эти подхода.
В примере ниже двумерный массив представляется в виде одномерного, а местоположения каждого элемента двумерного массива в одномерном определяется суммой номера столбца и произведения номера строки на длину строки.
Задача. Пример программы для работы с динамическим двумерным массивом как с одномерным.
void vvodMatr (int *matr, int n, int m) {
int i, j;
for(i=0; i<n; i++)
for(j=0; j<m; j++)
scanf("%d", &matr[i*m+j]); // matr+i*m+j==&matr[i*m+j]
}
void main() {
int n, m, i, j;
int *a; // указатель для массива a[n][m]
scanf("%d %d", &n, &m);
// выделение памяти под матрицу a[n][m]
a = (int *) malloc(n * m * sizeof(int));
if(!a) { // a == NULL
printf("Недостаточно памяти!\n"); return;
}
vvodMatr(a, n, m);
printf("\nМатрица --------------\n");
for(i=0; i<n; i++) {
for(j=0; j<m; j++)
printf("%5d ", *(a+i*m+j)); // a[i*m+j] == *(a+i*m+j)
printf("\n");
}
free(a);
}
Следующий вариант представления динамического двумерного массива позволяет использовать привычную индексацию двумерного массива. Массив представляется в виде динамического одномерного массива указателей на строки двумерного массива. Для создания двумерного массива вначале нужно выделить память под массив указателей на строки (на одномерные массивы), а затем каждый из этих указателей связать с динамически выделенными областями, куда запишутся элементы строк (одномерных массивов).
Задача. Пример программы для работы c динамическим двумерным массивом как с обычным двумерным массивом.
void vvodMatr (int *matr[], int n, int m) {
for(int i = 0; i < n; i++)
for(int j = 0; j < m; j++)
scanf("%d", &matr[i][j]);// &matr[i][j] == *(matr+i)+j
}
void freeMatr (int *matr[], int n) {
// освобождение строк матрицы
for(int i=0; i<n; i++)
free(matr[i]); // matr[i] == *(matr+i)
// освобождение массива указателей
free(matr);
}
void main() {
int n, m, i, j;
int **a; // матрица a[n][m]
scanf("%d %d", &n, &m);
// выделение памяти под матрицу
// выделение памяти под n указателей на строки матрицы
a = (int **) malloc( n * sizeof(int *) );
if(!a) { printf("Недостаточно памяти!\n"); return; }
// выделение памяти под строки матрицы
for(i = 0; i < n; i++) {
a[i] = (int *) malloc( m * sizeof(int) );
if(!a[i]) {
printf("Недостаточно памяти!\n");
freeMatr(a, i);
return;
}
}
vvodMatr(a, n, m);
printf("\nМатрица---------------\n");
for(i = 0; i < n; i++) {
for(j = 0; j < m; j++)
printf("%4d ", a[i][j]); // a[i][j] == *(*(a+i)+j)
printf("\n");
}
freeMatr(a, n);
}