Bootstrap

C++文件操作基础+实战

目录

C++文件操作理论

        1、打开文件用于读和写 open()

        2、检查文件打开是否成功 fail()

        3、读或者写 read(),write()

        4、检查是否读完 EOF(end of file)

        5、使用完文件后关闭文件 close()

        6、文件的打开方式

C++文件操作基础实战

        1、以追加数据的方式打开文件test_demo.txt,向该文件写入我们输入的数据

        2、将文件test_demo.txt中的内容拷贝到文件test_demo2.txt 中


C++文件操作理论

C++语言把文件看作是一个有序的字节序列,每个文件都以文件结束标志结束。一个文件是一个字符流或二进制流

C++文件不是由记录组成,它把数据看作是一连串的字符,输入时回车换行符作为符号同时被读入,输出时不会自动增加回车换行符作为标志。这种以字节流或二进制流组成的文件被称为流式文件

按照文件中的数据的组织形式可把文件分成为:

1)文本文件:文件中信息形式为ASCII码文件,每个字符占一个字节;

2)二进制文件:文件中信息的形式与其在内存中的形式相同。

注意:用ASCII形式输出与字符一一对应,一字节代表一个字符;用二进制形式输出数据,一字节并不对应一个字符,不能直接输出字符形式,但可以节省外存空间和转换空间

文件操作步骤,对于文件操作要做以下事情:

        1、打开文件用于读和写 open()

                建立流对象,调用open()函数连接外部文件

                格式

                        流类 对象名;

                        对象名.open("文件名",方式);

                “流类”是流类库中定义的文件流类,ifstream用于以读方式打开;ofstream用于以写方式         打开文件;fstream用于以读/写方式打开文件。“方式”是ios定义的表示常量,表示文件的打开

	fstream fout;
	fout.open("test_demo.txt",ios::app);//以追加数据的方式打开文件

                注意:这里用open()函数连接外部文件时,如果不设置打开方式,该函数默认的是以  ios::in | ios::out 这种方式打开 ,若外部文件不存在则会报错。下面是open()函数的定义,大家看一下就一目了然了。

	void open(const char *_Filename,
		ios_base::openmode _Mode = ios_base::in | ios_base::out,
		int _Prot = (int)ios_base::_Openprot)
		{	// open a C stream with specified mode
		if (_Filebuffer.open(_Filename, _Mode, _Prot) == 0)
			_Myios::setstate(ios_base::failbit);
		else
			_Myios::clear();	// added with C++11
		}

	void open(const string& _Str,
		ios_base::openmode _Mode = ios_base::in | ios_base::out,
		int _Prot = (int)ios_base::_Openprot)
		{	// open a C stream with specified mode
		open(_Str.c_str(), _Mode, _Prot);
		}

	void open(const char *_Filename, ios_base::open_mode _Mode)
		{	// open a C stream with specified mode (old style)
		open(_Filename, (ios_base::openmode)_Mode);
		}

        2、检查文件打开是否成功 fail()

                检查文件打开是否成功,这里有两种方法,一种是用流对象名调用ofstream中的fail()来判断,另一种是直接判断流对象是否为空,为空的话则表示文件打开失败。

                  格式:

                            流对象名.fail();

	//if (!fout) //检查文件是否打开成功
	//{
	//	cout << "open file faild!" << endl;
	//}
	if (fout.fail()) //检查文件是否打开成功
	{
		cout << "open file faild!" << endl;
	}

        3、读或者写 read(),write()

          首先,我们来看一下ifstream中read()函数的定义:

	_Myt& __CLR_OR_THIS_CALL read(_Elem *_Str, streamsize _Count)
		{	// read up to _Count characters into buffer
		ios_base::iostate _State = ios_base::goodbit;
		_Chcount = 0;
		const sentry _Ok(*this, true);

		if (_Ok && 0 < _Count)
			{	// state okay, use facet to extract
			_TRY_IO_BEGIN
			_DEBUG_POINTER(_Str);
			const streamsize _Num = _Myios::rdbuf()->sgetn(_Str, _Count);
			_Chcount += _Num;
			if (_Num != _Count)
				_State |= ios_base::eofbit | ios_base::failbit;	// short read
			_CATCH_IO_END
			}

		_Myios::setstate(_State);
		return (*this);
		}

         这里有两个参数:

                 _Elem *_Str:保存我们读取到的信息,_ELem是类模板basic_istream的一个类型参数,程序编译时,由编译器根据调用语句中实参的类型对类模板实例化,用实际数据类型替换类型参数_Elem。这里我们定义的是一个字符数组char temp[bufferlen],所以_ELem的类型是char,*_Str指针指向该数组的首地址

                 streamsize _Count:每一次读取到的最大的字节数

         格式:

                  ifstream的流对象名.read();

         接下来,我们来看一下ostream中write()函数的定义:

	_Myt& __CLR_OR_THIS_CALL write(const _Elem *_Str,
		streamsize _Count)
		{	// insert _Count characters from array _Str
		ios_base::iostate _State = ios_base::goodbit;
		const sentry _Ok(*this);

		if (!_Ok)
			_State |= ios_base::badbit;
		else if (0 < _Count)
			{	// state okay, insert characters
			_DEBUG_POINTER(_Str);
			_TRY_IO_BEGIN
			if (_Myios::rdbuf()->sputn(_Str, _Count) != _Count)
				_State |= ios_base::badbit;
			_CATCH_IO_END
			}

		_Myios::setstate(_State);
		return (*this);
		}

           这里也有两个参数,它们的意思跟上面 read()函数中的那两个参数意思差不多 ,我简单叙述一下:     

          _Elem *_Str:写入我们读取到的信息

          streamsize _Count:每一次写入的最大的字节数  

         格式:

                    ofstream的流对象名.write();

	//从源文件读取数据,写到目标文件当中
	//通过读取源文件的EOF来判断是否读取结束
	char temp[bufferlen];
	while (!in.eof())
	{
		//担心缓冲区的内存不够,所以一次性只读取bufferlen个字符,
	   //当某一次读取的字符达不到bufferlen个,则返回实际读到的字符
		in.read(temp, bufferlen); 
		//从每一次读取字符的过程中,计算我们实际获取的bufferlen的长度
		//从最后一次读取字符的过程中,若读取的字符达不到bufferlen个
		//则计算我们实际获取的bufferlen的长度
		streamsize count=in.gcount();
		out.write(temp, count);//这里会把文件结束标志一并写入
	}

         注意:这里用到了gcount()函数的方法,从注释上可以看到,它最大的好处在于从最   后一次读取字符的过程中,若读取的字符达不到bufferlen个,则计算我们实际获取的bufferlen的长度

        4、检查是否读完 EOF(end of file)

                这个比较简单,因为是读,所以直接用ifstream的对象名调用即可,它的作用是判断是                    否读到了文件尾

                  格式:

                            流对象名.eof();

        5、使用完文件后关闭文件 close()

                关闭文件操作主要是将缓冲区数据完整的写入文件,添加文件结束标志,使文件流与对                  应的物理文件断开联系

                  格式:

                          流对象名.close();

                因为这里我们用到了读和写,所以要分别用相应的流对象名去调用close()

	//关闭源文件和目标文件
	in.close();
	out.close();

       6、文件的打开方式

方式选项说明
ios::in打开文件进行读操作(ifstream默认模式),文件不存在时出错
ios::out打开文件进行写操作(ofstream默认模式),如文件已存在则更新文件
ios::ate

打开一个已有输入或输出的文件,文件指针定位到该文件的文件尾,若程序移动了文件指针,就把文件写到当前位置

ios::app打开文件以便在文件的尾部添加数据
ios::nocreate打开一个已经存在的文件,如果文件不存在则打开失败
ios::trunc若文件存在,则清除文件原有内容(默认)
ios::noreplace打开一个不存在的文件,如文件存在则打开失败
ios::binary以二进制文件方式打开(非文本文件)

C++文件操作基础实战

        1、以追加数据的方式打开文件test_demo.txt,向该文件写入我们输入的数据

源代码如下:

// demo.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include<iostream>
#include<fstream>
using namespace std;


int main()
{
	int x, index = 0;
	static int i = 1;
	fstream fout;
	fout.open("test_demo.txt",ios::app);//以追加数据的方式打开文件
	if (fout.fail()) //检查文件是否打开成功
	{
		cout << "open file faild!" << endl;
	}
	while (cin >> x)
	{
		fout << "第" << i << "个数据是" << x << endl;
		index++;
		i++;
		if (index == 5)
		{
			break;
		}
	}
	cin.ignore(numeric_limits<std::streamsize>::max(), '\n');//清楚缓存区的脏数据
	char ch;//定义最后一个数据的类型
	cin >> ch;
	fout << "最后一个数据是:" << ch << endl;
	fout.close();
    return 0;
}

这里可以看到我们输入的数据已成功保存至该文档 

      2、将文件test_demo.txt中的内容拷贝到文件test_demo2.txt 中

源代码如下:

// demo.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include<iostream>
#include<fstream>
using namespace std;

static const int bufferlen = 2048;
bool Copyfile(const string&src, const string&dst)
{
	//以读方式打开源文件
	ifstream in(src.c_str(), std::ios::in);
	//以写方式打开目标文件
	ofstream out(dst.c_str(), std::ios::out|std::ios::trunc);
	//判断文件是否打开成功,失败则返回false
	if (!in || !out)
	{
		return false;
	}
	//从源文件读取数据,写到目标文件当中
	//通过读取源文件的EOF来判断是否读取结束
	char temp[bufferlen];
	while (!in.eof())
	{
		//担心缓冲区的内存不够,所以一次性只读取bufferlen个字符,
	   //当某一次读取的字符达不到bufferlen个,则返回实际读到的字符
		in.read(temp, bufferlen); 
		//从每一次读取字符的过程中,计算我们实际获取的bufferlen的长度
		//从最后一次读取字符的过程中,若读取的字符达不到bufferlen个
		//则计算我们实际获取的bufferlen的长度
		streamsize count=in.gcount();
		out.write(temp, count);//这里会把文件结束标志一并写入
	}
	//关闭源文件和目标文件
	in.close();
	out.close();

	return true;
}

int main()
{
	cout << Copyfile("test_demo.txt", "test_demo2.txt") << endl;
	return 0;
}

可以看到,拷贝成功!

友情提示:实战部分的代码在理论部分已讲解完毕 

;