- •А. Нейбауэр.
- •Глава 1. Основы программирования 11
- •Глава 3. Переменные и константы 39
- •Глава 6. Операторы 103
- •Глава 13. Как собрать все вместе 253
- •Благодарности
- •Введение
- •Что вам даст эта книга
- •Программные средства
- •Глава 1. Основы программирования
- •Компьютерная программа
- •Языки программирования
- •Компиляторы
- •Интерпретатор
- •Скорость
- •Переносимость
- •Структурирование
- •Библиотеки функций
- •Необходимые пояснения
- •Что такое объектно-ориентированное программирование
- •Что Си может и чего не может
- •Этапы программирования
- •План программы
- •Текст программы
- •Компиляция программы
- •Компоновка программы
- •Тестирование программы
- •Изучение основ программирования
- •Что нужно, чтобы писать программы
- •Структура программы
- •Прописные и строчные символы
- •Инструкция return
- •Использование комментариев
- •Понятие параметров
- •Директива #include
- •Проектирование программы
- •Глава 3. Переменные и константы
- •Символьные данные
- •Целочисленные величины
- •Вещественные числа
- •Почему надо использовать целые числа?
- •Константы и переменные
- •Имена констант и переменных
- •Определение констант
- •Почему используют константы?
- •Определение переменных
- •Присваивание значения
- •Определение строковой переменной
- •Типы данных и функции
- •Литералы
- •Проектирование программы
- •Функция puts()
- •Функция putchar()
- •Двойственность символьных переменных
- •Управляющие коды
- •Код «новая строка»
- •Код «табуляция»
- •Код «возврат каретки»
- •Код «возврат на шаг»
- •Код «перевод страницы»
- •Отображение специальных символов на экране монитора
- •Многогранная функция printf()
- •Вывод чисел
- •Перевод строки
- •Преобразование типов данных
- •Форматированный вывод
- •Выбор правильных средств вывода информации
- •Проектирование программы
- •Функция gets()
- •Функция getchar()
- •«Для продолжения нажмите Enter»
- •Оператор получения адреса &
- •Функция scanf()
- •Входной поток
- •Использование функции scanf()
- •Выбор соответствующих средств ввода данных
- •Будьте осторожны при использовании scanf()
- •Неинициализированные переменные
- •Используемые алгоритмы ввода
- •Глава 6. Операторы
- •Арифметические операторы
- •Деление нацело
- •Типыданныхи операторы
- •Выражения
- •Приоритет операторов и порядок вычислений
- •Используемые алгоритмы обработки данных
- •Счетчики
- •Операторы инкремента
- •Аккумуляторы
- •Операторы присваивания
- •Присваивание начального значения
- •Проектирование программы
- •Остерегайтесь логических ошибок
- •Ищите образцы
- •Диагностические проблемы
- •Глава 7. Для чего нужны функции
- •Как использовать функции
- •Переменные в функциях
- •Автоматические (локальные) переменные
- •Внешние (глобальные) переменные
- •Статические переменные
- •Передача параметров
- •Возвращаемые значения
- •Возврат значений типа float
- •Использование return() в функции main()
- •Использование макроопределений
- •Проектирование программы
- •Автоматические или внешние переменные?
- •Неправильный ввод
- •Глава 8. Позвольте компьютеру принимать решения
- •Условия
- •Составные инструкции
- •Конструкция if...Else
- •Дополненный Опросник
- •Логические операторы
- •Вложенные инструкции if
- •Конструкция switch/case/default
- •Проверка чисел с плавающей точкой и строк
- •Проектирование программы
- •Проверка правильности ввода
- •Глава 9. Циклы
- •Использование цикла for
- •Создание паузы в программе
- •Составные инструкции
- •Использование переменных
- •Вложенные циклы
- •Использование цикла do...While
- •Вложенные циклы do
- •Использование цикла while
- •Комбинирование циклов разных типов
- •Проектирование программы
- •Использование флагов
- •Использование инструкции break
- •Глава 10. Массивы и строки
- •Массивы
- •Определение массива
- •Ввод значений в массив
- •Работа с массивами
- •Просмотр массива
- •Поиск в массиве
- •Передача массива функции
- •Использование массивов
- •Сравнение двух строк
- •Определение длины строки
- •Присваивание строк
- •Слияние строк
- •Массивы строк
- •Проектирование программы
- •Глава 11. Структуры и указатели
- •Использование структур
- •Определение структуры
- •Определение структурных переменных
- •Присвоение начального значения
- •Использование структуры
- •Массивы структур
- •Структуры и функции
- •Указатели
- •Указатели и функции
- •Глава 12. Вывод на диск и принтер
- •Что такое файловая структура
- •Указатель на файл
- •Как открыть файл
- •Как избежать ошибок выполнения
- •Как закрыть файл
- •Функции ввода и вывода
- •Работа с символами
- •Посимвольное чтение из файла
- •Работа со строками
- •Чтение строк
- •Форматированный ввод и вывод
- •Чтение форматированных файлов
- •Работа со структурами
- •Чтение структур
- •Чтение в массив
- •Дополнение файла новыми данными
- •Текстовый и двоичный форматы
- •Двоичный формат
- •Печать данных
- •Проектирование программы
- •Глава 13. Как собрать все вместе
- •Прикладная программа
- •Глобальные определения
- •Функция main()
- •Добавление записей: функция addcd()
- •Удаление записи: функция delcd()
- •Редактирование данных: функция chcd()
- •Изменение номера ячейки: функция chloc()
- •Вывод записи на экран: функция locate()
- •Печать записей: функция plist()
- •Сортировка записей: функция sort()
- •Приложение I Ответы и решения Глава 1. Основы программирования.
- •Глава 3. Переменные и константы.
- •Глава 6. Операторы.
- •Глава 7. Для чего нужны функции.
- •Глава 8. Позвольте компьютеру принимать решения.
- •Глава 9. Циклы.
- •Глава 10. Массивы и строки.
- •Глава 11. Струтуры и указатели.
- •Глава 12. Вывод на диск и принтер.
- •Приложение II Прикладная программа
Добавление записей: функция addcd()
Функция, которая добавляет информацию о компакт-диске, построена по шаблону, аналогичному тому, который использовался в главе 12 в программе добавления данных в файл.
Между примером, показанным в главе 12, и функцией, использующейся в данной программе, есть, тем не менее, три отличия. В нашей программе пользователь не может добавить информацию о новом диске, если в контейнере нет места, куда бы можно было поместить этот диск. Функция getslots() подсчитывает количество дисков, уже имеющихся в контейнере. Когда их общее число достигает максимального, на экране появляется соответствующее сообщение и программа возвращается в меню:
int pause;
if (count >= MAX)
{
puts("К сожалению, свободных ячеек нет\n");
pause = getchar();
return;
}
Далее, в нашей программе файл открывается с режимом доступа "a", поэтому файл будет создан даже в том случае, если он не существовал к моменту запуска программы:
if ((fp = fopen(FILENAME, "a")) == NULL)
{
printf("Невозможно открыть файл %s\n", FILENAME);
exit();
}
Если файл успешно открыт или вновь создан, данные будут добавлены в его конец.
Следующая функция просит пользователя ввести номер ячейки от1 до20. Так как ввод этой информации необходим при выполнении нескольких задач в программе, мы выделили его в отдельную функцию, которая называется getslot() и может быть вызвана при возникновении в этом необходимости:
getslot()
{
int index, flag, pause;
do
{
flag = 0;
printf("Введите номер ячейки: ");
scanf("%d", &disc.number);
for (index = 0; index < count; index++)
{
if (slots[index] == disc.number)
{
printf("К сожалению, ячейка уже занята, попробуйте другую\n");
flag = 1;
}
}
}
while (disc.number < 1 || disc.number > MAX || flag == 1);
count++;
slots[count] = disc.number;
return;
}
Функция getslot() не позволяет вводить номер уже занятой ячейки. После записи в файл информации о новом компакт-диске происходит вызов функции getslots(), которая обновляет массив номеров ячеек.
Удаление записи: функция delcd()
Функция, удаляющая информацию о диске, использует стандартный алгоритм для работы с последовательными файлами. Используя последовательный доступ, мы не можем непосредственно перейти к интересующему нас месту в файле, чтобы изменить содержащуюся там запись. Если открыть файл с режимом доступа "w", его содержимое будет уничтожено. При использовании режима доступа "a" мы сможем только добавить данные в конец файла.
Решение этой проблемы лежит в использовании двух файлов. Функция delcd() открывает файл с данными, используя режим доступа "r", и временный файл с режимом доступа "w":
if (fp = fopen(FILENAME, "r")) == NULL)
{
printf("Невозможно открыть файл %s\n", FILENAME);
exit();
}
if ((tp = fopen(tempfile, "w")) == NULL)
{
printf("Невозможно открыть файл %s\n", tempfile);
fclose(fp);
exit();
}
После этого мы вводим название диска, который следует удалить:
printf("Введите название диска: ");
gets(delname);
Цикл while осуществляет чтение каждой структуры (записи) из исходного файла:
while (fread(&disc, sizeof(disc), 1, fp) == 1)
{
Если запись не является той, которую пользователь хочет удалить, функция delcd() заносит ее во временный файл:
if (strcmp(disc.name, delname) != 0)
fwrite(&disc, sizeof(disc), 1, tp);
В нашей программе функция strcmp() используется в нескольких местах для определения того, прочитана ли из файла нужная запись. Функция strcmp() не входит в K&R-стандарт Си, но она есть во всех библиотеках ANSI Си и Си++.
Встретив нужную запись, функция delcd() не копирует ее во временный файл, а вместо этого устанавливает флаг, чтобы впоследствии узнать, была ли структура удалена:
else
fflag = 'y';
}
Цикл while продолжает копирование из одного файла в другой до тех пор, пока не достигнет конца файла, после чего оба файла закрываются:
fclose(fp);
fclose(tp);
Если функция delcd() не нашла записи, которую пользователь пожелал удалить, она выведет на экран соответствующее сообщение, после чего завершит работу:
puts("\nДиск с таким названием не найден");
pause = getchar();
Произведя удаление записи, вы получите два разных файла. Исходный файл содержит все данные, включая и информацию, которую вы хотите удалить. Временный файл содержит только данные, которые требовалось сохранить, так как функция delcd() не записала в него структуру, подлежащую удалению. Но мы не можем оставить данные в таком виде, программа способна вносить дополнительные данные и выполнять другие функции лишь с исходным файлом, поэтому мы должны привести его содержимое в соответствие с полученным результатом.
Некоторые компиляторы Си++ поставляются с библиотечными функциями, позволяющими измененять имена файлов, но в некоторых компиляторах они отсутствуют. В общем виде мы должны снова открыть оба файла, теперь уже с обратным порядком доступа, то есть исходный файл открывается с режимом доступа "w", после чего данные из него будут удалены, а временный файл открывается с режимом доступа "r". Для этого используется функция openwr(), вызываемая функцией delcd() и другими функциями:
if (fp = fopen(FILENAME, "w")) == NULL)
{
printf("Невозможно открыть файл %s\n", FILENAME);
exit();
}
if ((tp = fopen(TEMPFILE, "r")) == NULL)
{
printf("Невозможно открыть файл %s\n", tempfile);
fclose(fp);
exit();
}
После того как файлы открыты, функция delcd() считывает записи из временного файла и переносит их в исходный файл. После закрытия файлов, исходный файл содержит уже исправленную информацию, не включающую данные, которые мы хотели удалить:
while (fread(&disc, sizeof(disc), 1, tp) == 1)
fwrite(&disc, sizeof(disc), 1, fp);
fclose(fp);
fclose(tp);
Перед завершением функции delcd() программа вызывает функцию getslots(), обновляющую массив номеров ячеек.
Итак, теперь у нас есть два файла, которые содержат одинаковые данные. Единственный недостаток этой процедуры состоит в том, что диск компьютера должен иметь достаточно свободного места для хранения двух файлов с данными. Временный файл можно рассматривать как резервную копию. Если с исходным файлом произойдет что-то нежелательное, останется временный файл, содержащий данные в том виде, в каком они были до выполнения последней операции. Если вы по каким-либо причинам не хотите хранить временный файл на диске, то можете открыть его с режимом доступа "w", а потом немедленно закрыть:
tp = fopen(tempfile, "w");
fclose(tp);
В результате файл останется на диске, но совершенно пустой, так что он займет очень мало дискового пространства.
Программа продолжает проверку картотеки даже после того, как искомая запись удалена. Таким образом, вы можете удалить несколько одноименных записей.
|
Можно несколько усилить процедуру, если выводить на экран указанную запись и просить пользователя дать подтверждение, что эту запись действительно необходимо удалить. Если пользователь решит не удалять запись, программа должна скопировать ее во временный файл.