open是linux下的底层系统调用函数,fopen与freopen是c/c++的标准I/O库函数,带输入/输出缓冲。linxu下的fopen是open的封装函数,fopen最终还是要调用底层的系统调用open。所以在linux下如果需要对设备进行明确的控制,使用底层系统调用(open)
open对应的文件操作有:close, read, write,ioctl 等。
fopen 对应的文件操作有:fclose, fread, fwrite, freopen, fseek, ftell, rewind等。
open和fopen的区别:
1,fread是带缓冲的,read不带缓冲.
2,fopen是标准c里定义的,open是POSIX中定义的.
3,fread和fwrite参数为文件指针,read和write为文件描述符
4,fopen不能指定要创建文件的权限.open可以指定权限.
5,fopen返回文件指针,open返回文件描述符(整数).
6,linux/unix中任何设备都是文件,都可以用open,read.
一、C
1.fopen,fdopen
std::FILE* fopen( const char* filename, const char* mode );
FILE *fdopen(int fd, const char* mode);
1、fopen函数:打开文件filename
并返回与该文件关联的文件流。mode
用于确定文件访问模式。
2、fdopen函数:打开已存在的文件描述符,使标准I/O流与该文件相结合。主要用于fopen不能打开的特殊文件(如管道和网路通信等),这时必须先调用设备专用函数获得一个文件描述符,然后用fdopen使一个标准I/O与该文件描述符相结合。
参数
filename | - | 将文件流关联到的文件名 |
mode | - | 以空字符结尾的字符串确定文件访问模式 |
返回值
如果成功,则返回一个指向控制已打开文件流的对象的指针,同时清除 eof 和错误位。出错时,返回一个空指针。
2.fclose
int fclose(FILE *stream)
C 库函数 int fclose(FILE *stream) 关闭流 stream。刷新所有的缓冲区。
参数
stream 是指向 FILE 对象的指针,该 FILE 对象指定了要被关闭的流。
返回值
如果流成功关闭,则该方法返回零。如果失败,则返回 EOF。
3.FILE结构体
C语言的stdio.h头文件中定义了用于文件操作的结构体FILE。通过fopen返回一个文件指针(指向FILE结构体的指针)来进行文件操作。可以在stdio.h头文件中查看FILE结构体的定义,如下:
不过在visual studio中貌似被隐藏了,见https://docs.microsoft.com/en-us/cpp/porting/visual-cpp-change-history-2003-2015
4.fprintf()和fscanf()函数
int fprintf(FILE *stream, const char *format, ... );
fprintf()函数根据指定的format(格式)发送信息(参数)到由stream(流)指定的文件.
int fscanf(FILE *stream, const char *format, ...)
C库函数 int fscanf(FILE *stream, const char *format, ...) 从流stream读取格式化输入。
- stream -- 指向 FILE 对象的指针,该 FILE 对象标识了流。
- format -- C 字符串,包含了以下各项中的一个或多个:空格字符、非空格字符 和 format 说明符。
成功则返回成功赋值或写入的个数,失败或到达文件末尾返回负数。
5.fread和fwrite
这两个函数对应fopen.
fread()
头文件:#include<stdio.h>size_t fread(void*buffer, size_t size, size_t count, FILE*stream);
1.buffer: 读取的数据存放的内存的指针,(可以是数组,也可以是新开辟的空间)
指向缓冲区保存或读取的数据或者是用于接收数据的内存地址
2.size: 每次读取的字节数
3.count: 读取的次数
4.stream: 是要读取的文件的指针,是数据读取的流(输入流)
返回值:成功:实际读取的元素(并非字节)数目;失败:返回0
如果输入过程中遇到了文件尾或者出错,可能比请求的元素数目小fwrite()
头文件:#include<stdio.h>size_t fwrite(void*buffer, size_ size, size_t count, FILE*stream)
1.buffer:指向用于保存数据的内存位置的指针。对于fwrite来说,是要获取数据的地址
2.size: 每次读取的字节数
3.count: 读取的次数
4.stream: 数据写入的流(目标指针的文件)
返回值:实际写入的元素(并非字节)数目
如果输入过程中遇到了文件尾或者出错,可能比请求的元素数目小
二、C++
头文件fstream 定义了三个类型来支持文件IO: ifstream从一个给定文件读取数据,ofstream向一个给定文件写入数据,以及fstream可以读写给定文件。
每个流都有一个关联的文件模式(file mode),用来指出如何使用文件。表8.4列出了文件模式和它们的含义。
每个文件流类型都定义了一个默认的文件模式,当未指定文件模式时就使用此默认模式。与ifstream关联的文件默认以in模式打开,与ofstream关联的文件默认以out模式打开,与fstream关联的文件默认以in和out模式打开。
默认情况下,打开一个ofstream时,文件的内容会被丢弃。阻止一个ofstream清空给定文件内容的方法是同时指定app或in模式:
//在这几条语句中,filel都被截断
ofstream out ("filel"); //隐含以输出模式打开文件并截断文件
ofstream out2("file1", ofstream::out); //隐含地截断文件
ofstream out3("file1", ofstream::out | ofstream::trunc);
//为了保留文件内容,显式指定app模式
ofstream app("file2", ofstream::app); //隐含为输出模式
ofstream app2("fi1e2", ofstream::out | ofstream::app);
三、示例程序
从一个文件中读数据到另一个文件:
#include<iostream>
using namespace std;
int main()
{
FILE* fdRead = fopen("read.txt", "r");
FILE* fdwrite = fopen("write.txt", "a+");
char buffer[4];
int n = 0;
while (n = fread(buffer, sizeof(buffer), 1, fdRead))
{
cout <<"n = " << n << ' ';
int m = fwrite(buffer, sizeof(buffer), 1, fdwrite);
cout << "m = " << m << ' ';
cout << endl;
}
fclose(fdRead);
fclose(fdwrite);
return 0;
}
C/C++混合写法:
#include<iostream>
#include<string>
#include<algorithm>
#include<sstream>
#include<fstream>
#include<stdio.h>
using namespace std;
const int MAX_NUM = 100;
int a[MAX_NUM];
int n = 2;
int main(int argc, char* argv[])
{
FILE* file4 = fopen("d.txt", "w");
if (!file4) {
printf("file4 open error!\n");
return -1;
}
//fprintf(file4, "name age sex position\n");
int age, sex;
char name[50], position[50];
printf("please input 2 date : name age sex position:\n");
while (n--)
{
scanf("%s %d %d %s", &name, &age, &sex, &position);
fprintf(file4, "%s %d %d %s\n", name, age, sex, position);
}
fclose(file4);
FILE* file5 = fopen("d.txt", "r");
if (!file5) {
printf("file5 open error!");
return -1;
}
int m = 2, ans, ans1;
while (m--)
{
fscanf(file5, "%s %d %d %s", &name, &age, &sex, &position);
printf("%s %d %d %s\n", name, age, sex, position);
}
fclose(file5);
fstream file1("d.txt");
if (!file1) {
cout << "file1 open error! " << endl;
return -1;
}
string tmp;
vector<string>str;
while (file1 >> tmp)
{
str.push_back(tmp);
cout << tmp << endl;
}
ifstream file2("b.txt", ios::in);
if (!file2) {
cout << "file2 open error! " << endl;
return -1;
}
ofstream file3("c.txt", ios::out);
if (!file3) {
cout << "file3 open error! " << endl;
return -1;
}
while (n--)
{
file3 << n<<' ';
}
file1.close();
file2.close();
file3.close();
return 0;
}
输入输出结果如下: