Автор работы: Пользователь скрыл имя, 04 Декабря 2016 в 00:02, курсовая работа
Язык программирования функционирует для двух целей:
он предоставляет программисту аппарат для воплощения в реальность определенных действий;
он формирует концепции, используемые программистом, когда тот размышляет о своих будущих действиях.
fclose(fp);
return 0;
}
В данном примере задается специализированный указатель fp типа FILE, который инициализируется функцией fopen(). Функция fopen() в качестве первого аргумента принимает строку, в которой задан путь и имя файла. Вторым параметром определяется способ обработки файла, в данном случае, значение «w», которое означает открытие файла на запись с удалением всей прежней информации из него. Если файл открыт успешно, то указатель fp не будет равен NULL и с ним возможна работа. В этом случае с помощью функции putc() выполняется запись символов в файл, на который указывает указатель fp. Перед завершением программы открытый файл следует закрыть во избежание в нем потери данных. Это достигается функцией fclose(), которая принимает указатель на файл и возвращает значение 0 при успешном закрытии файла, иначе значение ЕОF.
Теперь рассмотрим пример программы считывания информации из файла.
Пример 2. Считывание текстовой информации из файла:
#include“stdafx.h”
#include“stdio.h”
int main(void)
{
setlocale(LC_ALL, “Russian”);
charstr_file[100];
FILE* fp= fopen(“my_file.txt”,”r”);
if(fp != NULL)
{
Int i =0;
charch;
while((сh = getc(fp)) != ЕОF)
str_file[i++]=ch;
str_file[i] = ‘\0’;
printf(str_file);
}
Else printf(“Невозможно открыть файл на чтение.\n”);
fclose(fp);
return 0;
}
В приведенном листинге функция
fopen() открывает файл на чтение,
что определяется значением второго аргумента
равною «r». Это значит, что в
него невозможно произвести запись данных,
а только считывание. Сначала
выполняется цикл while, в котором из файла
считывается символ с помощью функции
getc() и выполняется проверка: если считанное
значение не равно символу конца файла
ЕОF, то значение переменной сh записывается
в массив str_file. Данный цикл будет выполняться
до тех пор, пока не будут считаны все символы
из файла, т.е. пока не будет достигнут
символ EОF. После завершения цикла формируется
строка str_file, которая выводится на экран
с помощью функции printf(). Перед завершением
программы также выполняется функция
закрытия файла fclose().
Работа с текстовыми файлами через функции putc и getc не всегда удобна. Например, если необходимо записать или считать строку целиком, то желательно иметь функции, выполняющие эту работу. В качестве таковых можно воспользоваться функциями fputs() и fgets() для работы со строками. Перепишем предыдущие примеры с использованием данных функций.
Пример 3. Использование функций fputs() и fgets():
#include“stdafx.h”
#include“stdio.h”
int main(void)
{
charstr_file[]=”Строка для файла”;
FILE* fp = fopen(“mу_filе.txt”‚”w”);
if(fp != NULL)
fputs(str_file,fp);
else printf(“Невозможно открыть файл на запись.\n”);
fclose(fp);
fp = fopen(“mу_file.txt”,”r”);
if(fp != NULL)
{
fgets(str_file,sizeof(str_
printf(fp);
}
fclose(fp);
return 0;
}
Аналогичные Действия по записи данных в файл и считывания информации из него можно выполнить и с помощью функций fprintf() и fscanf(). Однако эти функции предоставляют большую гибкость в обработке данных файла. Продемонстрируем это на следующем примере. Допустим, имеется структура, хранящая информацию о книге: название, автор, год издания. Необходимо написать программу сохранения этой информации в текстовый файл и их считывания. Пример использования данных функций представлен в примере 4.
Пример 4. Использование функций fprintf() и fscanf():
#include“stdafx.h”
#include“stdio.h”
#define N 2
Struct book
{
char name[100];
char author[100];
int year;
}
books [N];
int main(void)
{
for(inti=0;i<N;i++)
{
scanf("%s",books[i].name);
scanf("%s",books[i].author);
scanf("%d",&books[i].year);
}
for(inti=0;i<N;i++)
{
puts(books[i].name);
puts(books[i].auth0r);
printf("%d\n",books[i].year);
}
FILE* fp = fopen(“mу_filе.txt”‚ ”w”);
for(inti=0;i<N;i++)
fprintf(fp,"%s %s %d\n",books[i].name,books[i].
fclose(fp);
fp =fopen("my7file.txt","r");
for(i=0;i<N;i++)
fscanf(fp,"%s %s %d\n“, books[i].name,books[i].author,
fclose(fp);
printf("---------------------—
for(i=0;i<N;i++)
{
puts(books[i].name);
puts(books[i].author);
printf{("%d\n"‚books[i].уеаr);
}
return 0;
}
При выполнении Данной программы вводится информация по книгам в массив структур books и выводится введенная информация на экран. Затем открывается файл my_file.txt на запись, в который заносится информация по книгам в порядке: наименование, автор, год издания. Так как число книг в данном случае равно двум, то выходной файл будет выглядеть следующим образом:
OneginPushkin 1983
OblomovGriboedov 1985
Затем, файл my_file.txt открывается на чтение и с помощью функции scanf() осуществляется считывание информации в элементы структуры. В заключении считанная информация выводится на экран монитора.
Представленный пример показывает возможность структурированной записи информации в файл и ее считывания. Это позволяет относительно просто сохранять разнородные данные в файле для их дальнейшего использования в программах.
При внимательном рассмотрении предыдущих примеров можно заметить, что функции считывания информации из файла «знают» с какой позиции следует считывать очередную порцию данных. Действительно, в последнем примере функция fscanf(), вызываемая в цикле, «знает» что нужно считать сначала первую строку из файла, затем вторую и т.д. И программисту нет необходимости задавать позицию для считывания данных. Все происходит автоматически. Вследствие чего появляется такая особенность работы? Дело в том, что у любого открытого файла в программе, написанной на С++ имеется указатель позиции (номера), с которой осуществляется считывание данных из файла. При открывании файла на чтение номер этой позиции указывает на начало файла. Поэтому функция fscanf(), вызванная первый раз, считывает данные первой строки. По мере считывания информации из файла, позиция сдвигается на число считанных символов. И функция fscanf() вызванная второй раз будет работать уже со второй строкой в файле. Несмотря на то, что указатель позиции в файле перемещается автоматически, в языке С++ имеются функции fscanf() и ftell(), позволяющие программно управлять положением позиции в файле.
Синтаксис данных функций следующий:
Int fseek( FILE *stream‚ long offset, int origin);
Longftell( FILE *stream );
где *stream – указатель на файл; pffset – смещение позиции в файле (в байтах); origin – флаг начального отсчета, который может принимать значения: SEEK_END – конец файла, SEEK_SET – начало файла; SEEK_CUR – текущая позиция. Последняя функция возвращает номер текущей позиции в файле.4
Рассмотрим действие данных функций на примере считывания символов из файла в обратном порядке.
Пример 5. Использование функций fscanf() и ftell():
#include “stdafx.h”
#include “stdio.h”
Int main(void)
{
FILE* fp = fopen("my_file.txt“, “\w”);
if(fp != NULL)
{
Fprint(fp)"Это пример использования функций fseek и ftell");
}
fclose(fp);
fp = fopen(“my_file.txt“,“r”);
if(fp != NULL)
{
charch;
fseek(fp, 0,SEEK_END);
long length = ftell(fp);
printf("1ength = %d\n",1ength);
for(inti = 1;i<= length;i++)
{
fseek(fp,-i, SEEK_END);
ch = getc(fp);
pu tchar(ch);
}
}
fclose(fp);
return 0;
}
В данном примере сначала создается файл, в который записывается строка «Это пример использования функций fseek и ftell». Затем этот файл открывается на чтение и с помощью функции fseek(fp,0,SEEK_END) указатель позиции помещается в конец файла. Это достигается за счет установки флага SEEK_END‚ который перемещает позицию в конец файла при нулевом смещении. В результате функция ftell(fp) возвратит число символов в открытом файле. В цикле функция fseek(fp‚–i‚SEEK_END) смещает указатель позиции на –i символов относительно конца файла, после чего считывается символ функцией getc(), стоящий на i-й позиции с конца.
Так как переменная i пробегает значения от 1 до length, то на экран будут выведены символы из файла в обратном порядке.
Таким образом, данные примеры показывают основы работы с текстовыми файлами в C++. В основе лежат такие функции, как fopen(), putc(), fclose(), fprintf(), fscanf() и мн. др.
Следует отметить, что во всех рассмотренных выше примерах функция fopen() в режимах «r» и «w» открывает текстовый файл на чтение и запись соответственно. Это означает, что некоторые символы форматирования текста, например, возврат каретки ‘\r’ не могут быть считаны как отдельные символы, их как бы не существует в файле, но при этом они там есть. Это особенность текстового режима файла. Для более «тонкой» работы с содержимым файлов существует бинарный режим, который представляет содержимое файла как последовательность байтов где все возможные управляющие коды являются просто числами. Именно в этом режиме возможно удаление или добавление управляющих символов недоступных в текстовом режиме. Для того чтобы открыть файл в бинарном режиме используется также функция fopen() с последним параметром равным «rb» и «wb» соответственно для чтения и записи. Продемонстрируем особенности обработки бинарного файла на примере подсчета числа управляющих символов возврата каретки ‘\г’ в файле, открытый в текстовом и бинарном режимах.5
Пример 6. Программа подсчета числа символов ‘\r’ в файле:
#include“stdafx.h”
#include“stdio.h”
int main(void)
{
FILE* fp = fopen("my_file.tхt","w");
if(fp ! = NULL)
{
fprintf(fp,“Это\пример использования\nбинарных файлов“);
}
fclose(fp);
charch;
intcnt = 0;
fp = fopen(“my_file.txt“,"r");
if(fp != NULL)
{
while((ch = getc(fp)) != EOF)
if(ch == '\r‘) cnt++;
}
fclose(fp);
printf("Текстовый файл: количество= %d\n“,cnt);
cnt=0;
fp = fopen("my_file.txt",”rb");
if(fp != NULL)
{
while((ch = getc(fp)) != EOF)
if(ch == '\r') cnt++;
}
fclose(fp);
printf("Бинарный файл: количество = %d\n”,cnt);
return 0;
}
Результат работы будет следующий:
Textfile: cnt = 0
Binary file: cnt = 2
Анализ полученных данных показывает, что при открытии файла в текстовом режиме, символы возврата каретки ‘\r’ не считываются функцией getc(), а в бинарном режиме доступны все символы.
Еще одной особенностью текстового формата файла является запись чисел в виде текста. Действительно, когда в предыдущих примерах выполнялась запись числа в файл с помощью функции fprintf(), например, года издательства книги, то число заменялось строкой. А когда она считывалась функцией fscanf(), то преобразовывалась обратно в число. Если мы хотим компактно представлять данные в файле, то числа следует хранить как числа, а не как строки. При этом целесообразно использовать бинарный режим доступа к файлу, т.к. будет гарантия, что любое записанное число не будет восприниматься, как управляющий символ и будет корректно считан из файла.
Для работы с бинарными файлами предусмотрены функции fread() и fwrite() со следующим синтаксисом:
size_tfread( void *buffer, size_t size, size_t count, FILE *stream );
где *buffer – указатель на буфер памяти, в который будут считываться данные из файла; size – размер элемента в байтах; count – число считываний элементов; *stream – указатель на файл.
Size_tfwrite( void *buffer, size_t size, size_t count, FILE *stream );
где *buffer – указатель на буфер памяти, из которого будут считываться данные в файл; size – размер элемента в байтах; сount – число записей; *stream – указатель на файл.6
Приведем пример использования функций fwrite() и fread(),
Пример 7. Использование функций fwrite() и fread():
#include“stdafx.h”
#include“stdio.h”
int main(void)
{
FILE *stream;
char list[30];
inti, numread, numwritten;
if( (stream = fopen( "fread.out", “wb” )) != NULL )
{
for ( i= 0; i< 25; i++)
list[i] = (char)('z' — i);
numwritten =fwrite( list, sizeof( char ), 25, stream );