Под понятием произвольного доступа к файлу подразумевается ряд различных моментов:
- можно произвольно обращаться к любой записи или любому байту в файле, в противоположность последовательному доступу, когда данные извлекаются или передаются в поток строго по очереди;
- в открытом файле можно произвольно чередовать операции чтения и записи;
- ввод-вывод с произвольным доступом является по преимуществу бесформатным.
Для позиционирования потока используются функции seekg() и seekp().
Разница между ними в том, что первая перемещает позицию чтения в потоке, а вторая устанавливает новую позицию в выходном потоке. Если используется поток типа fstream, и он открыт в режиме чтения-записи, то все равно, какую функцию применять для позиционирования.
Для определения текущей позиции tellg() и tellp(). Первая возвращает позицию чтения во входном потоке, а вторая – позицию в выходном потоке.
Примеры программ
Рассмотрим примеры использования функции работы с файлами и строками.
Пример 11.3 Поиск вхождения слова в файле
#include "stdafx.h"
#include <iostream>
#include <fstream>
#include <string>
using namespace std;
int main()
{
setlocale(LC_ALL, "Russian");
const int len = 81;
char word[len], line[len], end_word[] = "done";
//пересоздадим файл и откроем для чтения/записи
fstream f("lab11.txt", ios::in | ios::out | ios::trunc);
if (!f)
{
cout << "Ошибка открытия файла" << endl;
return -1;
}
//введем несколько строк для дальнейшего поиска в них
do
{
cin >> line;
//с файловым потоком можно работать как со стандартными cin и cout
f << line << endl;
}
//продолжаем пока не введем done
while (strcmp(line, end_word));
//сбросим файловый поток на начало
f.seekg(0);
cout << "Содержимое файла: " << endl;
//содержимое файла на экран
//выполняем пока не достигнем конца файла
while (!f.eof())
{
f >> line;
cout << line << endl;
}
cout << "Введите слово для поиска: ";
cin >> word;
//сбрасываем бит-признак конца файла eofbit
f.clear();
//сбросим файловый поток на начало
f.seekg(0);
//определим длину искомого слова
size_t l_word = strlen(word);
//счетчик вхождения слова
int wc = 0;
//читаем построчно и ищем слово word в строке
while (f.getline(line, len))
{
//поместим указатель p на начало строки
char *p = line;
//strstr возвращает указатель
// на элемент из строки p с которого начинается word
while (p = strstr(p, word))
{
//используем указатель не текущую позицию в строке
char *c = p;
//переместим p на символ сразу за концом слова
p += l_word;
//проверим стоит ли наше слово отдельно
//или это просто подстрока в другом (большем) слове
//проверим совпадает ли начало слова с началом строки
if (c!= line)
//проверим символ перед началом слова
//на принадлежность к разделителям
if (!ispunct(*(c-1)) &&!isspace(*(c-1)))
//начинается не с начала строки и
//не с разделителя => ищем дальше
continue;
//символы перед началом слова подходят
//проверяем символы за окончанием слова
//если это пробелы, символы пунктуации
//или конец строки => увеличиваем счетчик слов
if (ispunct(*p) || isspace(*p) || (*p == '\0'))
{
wc++;
cout << "Слово найдено" <<endl;
}
}
}
cout << "Количество вхождений слова: "<< wc << endl;
return 0;
}
Рассмотрим другой подход к решению предложенной выше задачи. В библиотеке C++ есть функция strtok, которая разбивает предложенную строку на лексемы в соответствии с заданным набором разделителей. При ее использовании нет необходимости выделять и проверять начало и конец слова. Потребуется лишь сравнит с искомым словом, слово, выделенное strtok. Для этого просто необходимо задать разделители.
Пример 11.4 Поиск вхождения слова в файле с помощью strtok
#include "stdafx.h"
#include <iostream>
#include <fstream>
#include <string>
using namespace std;
int main()
{
setlocale(LC_ALL, "Russian");
const int len = 81;
char word[len], line[len], end_word[] = "done";
//пересоздадим файл и откроем для чтения/записи
fstream f("lab11.txt", ios::in | ios::out | ios::trunc);
if (!f)
{
cout << "Ошибка открытия файла" << endl;
return -1;
}
//введем несколько строк для дальнейшего поиска в них
do
{
cin >> line;
f << line << endl;
}
//продолжаем пока не введем done
while (strcmp(line, end_word));
cout << "Введите слово для поиска: ";
cin >> word;
//сбрасываем бит-признак конца файла eofbit
f.clear();
//сбросим файловый поток на начало
f.seekg(0);
//Список разделителей
char delims[] = ",.!? /<>|()*:;\"";
//Указатель на начало слова
char *token;
//счетчик вхождения слова
int wc = 0;
//читаем построчно и ищем слово word в строке
while (f.getline(line, len))
{
//находим первый символ не из разделителей
//находим первый символ не из разделителей
token = strtok(line, delims);
//проверяем, остались ли еще слова
while (token!= NULL)
{
//strtok заменяет символ после разделителя
//на NULL поэтому можно сравнивать искомое и
//найденное слово
if (!strcmp(token, word))
wc++;
//Для поиска следующей лексемы в той же строке
//strtok необходимо передать NULL
token = strtok(NULL, delims);
}
}
cout << "Количество вхождений слова: "<< wc << endl;
return 0;
}
Пример 11.5 Произвольный доступ к файлу
// Открыть файл как двоичный сразу для ввода и вывода
// (создать новый, если отсутствует или перезаписать старый)
// Ввести элементы с клавиатуры
// Поменять знак четных элементов
#include "stdafx.h"
#include <iostream>
#include <fstream>
#include <iomanip>
using namespace std;
int main()
{
setlocale(LC_ALL, "Russian");
const int NP = 10;
const int IS = sizeof(int); //размер элемента int
int pt, i;
// Открытие файла для чтения/записи.
fstream fs("random.txt",
ios::binary | ios::in | ios::out | ios::trunc);
if (!fs)
{
cerr << "Ошибка открытия файла." << endl;
return 1;
}
// Первоначальная запись файла.
cout << "Начальные заняения:" << endl;
for (i = 0; i < NP; i++)
{
pt = i;
//Приводим pt к типу char* для нормальной работы write
fs.write((char*)&pt, IS);
cout << setw(4) << pt;
}
cout << endl << endl;
// Чтение файла от конца к началу.
cout << "Читаем из файла в обратном порядке:"<< endl;
for (i=0; i<NP; i++)
{
//Перемещаемся к i-му элементу с конца
fs.seekg(-(i + 1) * IS, ios::end);
//Приводим pt к типу char* для нормальной работы read
fs.read((char*)&pt, IS);
cout << setw(4)<< pt;
};
cout<< endl << endl;
// Переписать четные индексы.
for (i=1; i<NP/2; i++)
{
//перемещаемся к i-му элементу
fs.seekg(2 * i * IS);
//читаем i-ый элемент
fs.read((char*)&pt, IS);
//меняем значение на противоположное
pt = -pt;
//возвращаемся на шаг назад, к только что прочитанному элементу
int p = fs.tellg();
p -= IS;
fs.seekg(p);
//перезаписываем его
fs.write((char*)&pt, IS);
}
//выводим файл на экран
cout << "После перезаписи:"<<endl;
fs.seekg(0);
for (i=0; i<NP; i++)
{
fs.read((char*)&pt, IS);
cout << setw(4) << pt;
}
cout << endl;
fs.close();
return 0;
}
Замечание. Когда эта программа открывает уже существующий файл, он усекается до нулевой длины (т.е. все его данные теряются). Если необходимо работать с уже имеющимися в файле данными, нужно убрать бит ios::trunc из режима открытия потока.
Контрольные вопросы
1. Какие виды строк существуют в C++?
2. Как объявить C-строку?
3. Как осуществляется ввод-вывод строк?
4. Какие операции над строками вы знаете?
5. Перечислите операции над символами?
6. Перечислите стандартные потоки ввода-вывода?
7. Как создать файловый поток?
8. Какие режимы открытия файлов существуют?
9. Что такое двоичный режим ввода-вывода?
10. Что такое текстовый режим ввода-вывода?
11. Как осуществляется чтение символов и строк?
12. Как изменить текущую позицию в файле?
13. Как узнать текущую позицию в файле?
14. Для чего используется метод clear()?
Задание
1. Написать программу в соответствии с вариантом задания из пункта 5.
2. Отладить и протестировать программу.
3. Оформить отчёт.
Варианты заданий
Вариант 1
Написать программу, которая считывает из текстового файла три предложения и выводит их в обратном порядке.
Вариант 2
Написать программу, которая считывает текст из файла и выводит на экран только предложения, содержащие введенное с клавиатуры слово.
Вариант 3
Написать программу, которая считывает текст из файла и выводит на экран только строки, содержащие двузначные числа.
Вариант 4
Написать программу, которая считывает английский текст из файла и выводит на экран слова, начинающиеся с гласных букв.
Вариант 5
Написать программу, которая считывает текст из файла и выводит его на экран, меняя местами каждые два соседних слова.
Вариант 6
Написать программу, которая считывает текст из файла и выводит на экран только предложения, не содержащие запятых.
Вариант 7
Написать программу, которая считывает текст из файла и определяет, сколько в нем слов, состоящих не более чем из четырех букв.
Вариант 8
Написать программу, которая считывает текст из файла и выводит на экран только цитаты, то есть предложения, заключенные в кавычки.
Вариант 9
Написать программу, которая считывает текст из файла и выводит на экран только предложения, состоящие из заданного количества слов.
Вариант 10
Написать программу, которая считывает английский текст из файла и выводит на экран слова текста, начинающиеся и оканчивающиеся на гласные буквы.
Вариант 11
Написать программу, которая считывает текст из файла и выводит на экран только строки, не содержащие двузначных чисел.
Вариант 12
Написать программу, которая считывает текст из файла и выводит на экран только предложения, начинающиеся с тире, перед которым могут находиться только пробельные символы.
Вариант 13
Написать программу, которая считывает английский текст из файла и выводит его на экран, заменив каждую первую букву слов, начинающихся с гласной буквы, на прописную.
Вариант 14
Написать программу, которая считывает текст из файла и выводит его на экран, заменив цифры от 0 до 9 на слова «ноль», «один»,..., «девять», начиная каждое предложение с новой строки.
Вариант 15
Написать программу, которая считывает текст из файла, находит самое длинное слово и определяет, сколько раз оно встретилось в тексте.
Вариант 16
Написать программу, которая считывает текст из файла и выводит на экран сначала вопросительные, а затем восклицательные предложения.
Вариант 17
Написать программу, которая считывает текст из файла и выводит его на экран, после каждого предложения добавляя, сколько раз встретилось в нем введенное с клавиатуры слово.
Вариант 18
Написать программу, которая считывает текст из файла и выводит на экран все его предложения в обратном порядке.
Вариант 19
Написать программу, которая считывает текст из файла и выводит на экран сначала предложения, начинающиеся с однобуквенных слов, а затем все остальные.
Вариант 20
Написать программу, которая считывает текст из файла и выводит на экран предложения, содержащие максимальное количество знаков пунктуации.
Вариант 21
Написать программу, которая считывает текст из одного файла, содержащего только цифры. Выводит его содержимое на экран, и записывает в другой файл заменив цифры от 0 до 9 на слова «ноль», «один»,..., «девять», начиная каждое число с новой строки, а цифры разделяя пробелами.
Вариант 22
Написать программу, которая считывает текст из файла и записывает в другой только строки, не содержащие символов пунктуации (точка, запятая и т.д.)
Вариант 23
Написать программу, которая считывает текст из файла и определяет, сколько в нем строчек, содержащих не более чем три пустых символа (пробел, табуляция и.т.д.).
Вариант 24
Написать программу, которая считывает из одного текстового файла строчки и сохраняет их в другой файл в обратном порядке.
Вариант 25
Написать программу, которая считывает текст с экрана в файл и выводит его обратно на экран, заменяя арабские цифры римскими (представление чисел не учитывать, заменять только сами цифры).
Содержание отчета
1. Титульный лист.
2. Наименование и цель работы.
3. Краткое теоретическое описание.
4. Задание на лабораторную работу.
5. Схема алгоритма.
6. Листинг программы.
7. Результаты выполнения программы.