C中文件的openmode如下:
r | 只读 | 为输入打开一个文本文件 |
w | 只写 | 为输出打开一个文本文件 |
a | 追加 | 向文本文件尾添加数据 |
rb | 只读 | 为输入打开一个二进制文件 |
wb | 只写 | 为输出打开一个二进制文件 |
ab | 追加 | 向二进制文件尾添加数据 |
r+ | 读写 | 为读写打开一个文本文件 |
w+ | 读写 | 为读写新建一个新的文本文件 |
a+ | 读写 | 为读写打开一个文本文件,并在文件尾进行写入 |
rb+ | 读写 | 为读写打开一个二进制文件 |
wb+ | 读写 | 为读写新建一个二进制文件 |
ab+ | 读写 | 为读写打开一个二进制文件,并在文件尾进行写入 |
/*
下面的函数是在vs中单步调试时得到的,具体操作是在fp.open()这里打个断点,然后单步调试(F11)
这里体现了C中和C++打开文件的各种模式的对应关系,即mods和valid中的元素是一一对应的
*/
_CRTIMP2_PURE FILE *__CLRCALL_PURE_OR_CDECL _Fiopen(const _Sysch_t *filename,
ios_base::openmode mode, int prot)
{ // open a file with native name
static const _Sysch_t *mods[] =
{ // fopen mode strings corresponding to valid[i]
_SYSCH("r"), _SYSCH("w"), _SYSCH("w"), _SYSCH("a"),
_SYSCH("rb"), _SYSCH("wb"), _SYSCH("wb"), _SYSCH("ab"),
_SYSCH("r+"), _SYSCH("w+"), _SYSCH("a+"),
_SYSCH("r+b"), _SYSCH("w+b"), _SYSCH("a+b"),
0}; static const int valid[] =
{ // valid combinations of open flags
ios_base::in,
ios_base::out,
ios_base::out | ios_base::trunc,
ios_base::out | ios_base::app,
ios_base::in | ios_base::binary,
ios_base::out | ios_base::binary,
ios_base::out | ios_base::trunc | ios_base::binary,
ios_base::out | ios_base::app | ios_base::binary,
ios_base::in | ios_base::out,
ios_base::in | ios_base::out | ios_base::trunc,
ios_base::in | ios_base::out | ios_base::app,
ios_base::in | ios_base::out | ios_base::binary,
ios_base::in | ios_base::out | ios_base::trunc
| ios_base::binary,
ios_base::in | ios_base::out | ios_base::app
| ios_base::binary,
0}; FILE *fp = 0;
int n;
ios_base::openmode atendflag = mode & ios_base::ate;
ios_base::openmode norepflag = mode & ios_base::_Noreplace; if (mode & ios_base::_Nocreate)
mode |= ios_base::in; // file must exist
if (mode & ios_base::app)
mode |= ios_base::out; // extension -- app implies out mode &= ~(ios_base::ate | ios_base::_Nocreate | ios_base::_Noreplace);
for (n = 0; valid[n] != 0 && valid[n] != mode; ++n)
; // look for a valid mode if (valid[n] == 0)
return (0); // no valid mode
else if (norepflag && mode & (ios_base::out || ios_base::app)
&& (fp = _Xfsopen(filename, _SYSCH("r"), prot)) != 0)
{ // file must not exist, close and fail
fclose(fp);
return (0);
}
else if (fp != 0 && fclose(fp) != 0)
return (0); // can't close after test open
else if ((fp = _Xfsopen(filename, mods[n], prot)) == 0)
return (0); // open failed if (!atendflag || fseek(fp, 0, SEEK_END) == 0)
return (fp); // no need to seek to end, or seek succeeded fclose(fp); // can't position at end
return (0);
}
其中的mods和valid是对应的,分别代表C和C++中文件的openmode,即
下面看c++中文件操作的openmode
ios::in |
打开文件进行读操作,如果文件不存在,则打开失败 |
ios::out |
打开文件进行写操作,如果文件不存在,则自动创建新文件[注1、注2] |
ios::app |
打开文件进行写操作,如果文件不存在,则自动创建新文件(这是因为ios::app相当与ios::app|ios::out,注3),但每次写操作之前,都会将指针放到文件末尾(意味着seekg/seekp无效,只能在文件尾写入数据) |
ios::ate | 打开文件时,指针定位到文件末尾(但可以通过seekg/seekp来调节指针位置,从而在指定位置写入数据) |
ios::trunc | 打开文件时清空原有数据 |
ios::binary |
以二进制形式打开文件,如未设置,则默认以文本形式打开文件 |
注:
1.只设置ios::out相当于设置了ios::out|ios::trunc。在代码中体现为_SYSCH("w"), _SYSCH("w")对应的分别是ios_base::out,ios_base::out | ios_base::trunc, _SYSCH("wb"), _SYSCH("wb")对应的分别是ios_base::out | ios_base::binary,ios_base::out | ios_base::trunc | ios_base::binary,对应着文本文件和二进制文件
2.ios::out的自动创建新文件的功能在只有ios::in|ios::out和ios::in|ios::out|ios::binary这两种情况下失效
3.ios::app相当于ios::app|ios::out,在代码中体现为:
if (mode & ios_base::app)
mode |= ios_base::out; // extension -- app implies out
最后,ofstream, ifstream 和 fstream所有这些类的成员函数open 都包含了一个默认打开文件的方式,这三个类的默认方式各不相同:
类 | 默认openmode |
ofstream | ios::out |
ifstream | ios::in |
fstream | ios::in | ios::out |
注:
1.只有当函数被调用时没有声明方式参数的情况下,默认值才会被采用。如果函数被调用时声明了任何参数,默认值将被完全改写,而不会与调用参数组合。
2.ofstream的默认openmode为ios::out,结合前面所讲,其实是ios::out | ios::trunc,且如果文件不存在会自动创建文件
3.fstream的默认openmode为ios::in | ios::out,结合前面所讲,此时不会清空文件,且如果不存在也不会自动创建文件