fopen

来源:互联网 发布:gta5卡怎么优化 编辑:程序博客网 时间:2024/05/29 19:13
函数原型:FILE * fopen(const char * path, const char * mode);
相关函数:open, fclose, fopen_s, _wfopen
返回值:文件顺利打开后,指向该流的文件指针就会被返回。如果文件打开失败则返回NULL,并把错误代码存在errno 中。

C语言

编辑

函数简介

函数原型:FILE * fopen(const char * path, const char * mode);
返回值:文件顺利打开后,指向该流的文件指针就会被返回。如果文件打开失败则返回 NULL,并把错误代码存在errno中。
一般而言,打开文件后会做一些文件读取或写入的动作,若打开文件失败,接下来的读写动作也无法顺利进行,所以一般在 fopen() 后作错误判断及处理。
参数说明:
参数 path字符串包含欲打开的文件路径及文件名,参数 mode 字符串则代表着流形态。
mode 有下列几种形态字符串:
字符串说明r以只读方式打开文件,该文件必须存在。r+以读/写方式打开文件,该文件必须存在。rb+以读/写方式打开一个二进制文件,只允许读/写数据。rt+以读/写方式打开一个文本文件,允许读和写。w打开只写文件,若文件存在则长度清为 0,即该文件内容消失,若不存在则创建该文件。w+打开可读/写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。a以附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留(EOF 符保留)。a+以附加方式打开可读/写的文件。若文件不存在,则会建立该文件,如果文件存在,则写入的数据会被加到文件尾后,即文件原先的内容会被保留(原来的 EOF 符不保留)。wb以只写方式打开或新建一个二进制文件,只允许写数据。wb+以读/写方式打开或建立一个二进制文件,允许读和写。wt+以读/写方式打开或建立一个文本文件,允许读写。at+以读/写方式打开一个文本文件,允许读或在文本末追加数据。ab+以读/写方式打开一个二进制文件,允许读或在文件末追加数据。
以 x 结尾的模式为独占模式,文件已存在或者无法创建(一般是路径不正确)都会导致 fopen 失败。文件以操作系统支持的独占模式打开。
上述的形态字符串都可以再加一个 b 字符,如 rb、w+b 或 ab+ 等组合,加入 b 字符用来告诉函数库以二进制模式打开文件。如果不加 b,表示默认加了 t,即 rt、wt,其中 t 表示以文本模式打开文件。由 fopen() 所建立的新文件会具有 S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH(0666) 权限,此文件权限也会参考umask值。
有些 C编译系统可能不完全提供所有这些功能,有的C版本不用"r+"、"w+"、"a+",而用"rw"、"wr"、"ar"等,读者注意所用系统的规定。
二进制和文本模式的区别
1、在Windows系统中,文本模式下,文件以"\r\n"代表换行。若以文本模式打开文件,并用 fputs 等函数写入换行符"\n"时,函数会自动在"\n"前面加上"\r"。即实际写入文件的是"\r\n"。
2、在类 Unix/Linux 系统中文本模式下,文件以"\n"代表换行。所以 Linux 系统中在文本模式和二进制模式下并无区别。
打开方式总结:各种打开方式主要有三个方面的区别
1、打开是否为二进制文件,用“b”标识。
2、读写的方式,有以下几种:只读、只写、读写、追加只写、追加读写这几种方式。
3、对文件是否必 须存在、以及存在时是清空还是追加会有不同的响应。具体判断如下图。

程序示例

示例一
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
#include <stdio.h>
#define F_PATH "d:\\myfile\\file.dat"
 
int main(void)
{
    FILE *fp = NULL; /* 需要注意 */
    fp = fopen(F_PATH, "r");
    if (NULL == fp)
    {
        return -1; /* 要返回错误代码 */
    }
    fclose(fp);
    fp = NULL; /* 需要指向空,否则会指向原打开文件地址 */
    return 0;
}
示例二
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
#include <stdio.h>
#include <stdlib.h> /* 为了使用exit() */
 
int main(void)
{
    int i = 0; /* 用于 putchar & getc 的数据接收 */
    char *ch = "";
    FILE *fp = NULL;
    char fname[50]; /* 用于存放文件名 */
    printf("输入文件名:");
    scanf("%s", fname);
    fp = fopen(fname, "r"); /* 只供读取 */
    if (NULL == fp) /* 如果失败了 */
    {
        printf("错误!");
        exit(1); /* 中止程序 */
    }
    while ((ch[i] = getc(fp)) != EOF)
    {
        putchar(ch[i]);
        i ++;
    }
    fclose(fp); /* 关闭文件 */
    fp = NULL; /* 需要指向空,否则会指向原打开文件地址 */
    return 0;
}
注意!初学者往往会犯一个错误,即在输入文件名时不加后缀名,请注意加上!
示例三
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
#include <stdio.h>
FILE *stream, *stream2;
 
int main(void)
{
    int numclosed; /* Open for read (will fail if file "crt_fopen.c" does not exist) */
    if ((stream = fopen("crt_fopen.c""r")) == NULL) /* C4996 */
        //Note: fopen is deprecated; consider using fopen_s instead
        printf("The file `crt_fopen.c' was not opened\n");
    else
        printf("The file `crt_fopen.c' was opened\n"); /* Open for write */
    if ((stream2 = fopen("data2""w+")) == NULL) /* C4996 */
        printf("The file `data2' was not opened\n");
    else
        printf("The file `data2' was opened\n");
    /* Closes tream if it is not NULL */
    if (stream)
    {
        if (fclose(stream))
        {
            printf("The file `crt_fopen.c' was not closed\n");
        }
    }
    /* All other files are closed: */
    numclosed = _fcloseall();
    printf("Number of files closed by _fcloseall: %u\n", numclosed);
}
示例四[1] 
Linux 下的程序示例。
在 / opt / C_lanuage / fopen_fread 新建两个文本,main.c 和 tmp.txt
tmp.txt:
I Love You Linux----Red Hat Enterprise----梦剧场的记忆
main.c 程序:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#include <stdio.h>
int main(void)
{
    FILE *fp = NULL;
    char tmp[100];
    fp = fopen("/opt/C_lanuage/fopen_fread/tmp.txt""r");
    if (NULL == fp)
    {
        printf("File open fail!\n");
        return -1;
    }
    fread(tmp, 1, 100, fp);
    printf("%s\n", tmp);
    fclose(fp);
    fp = NULL;
    return 0;
}
编译加执行
[root@localhost fopen_fread]# gcc - g main.c - o main
[root@localhost fopen_fread]# . / main
I Love You Linux----Red Hat Enterprise----梦剧场的记忆

注意

在文件操作时,需要注意以下几点问题:
1、在定义文件指针时,要将文件指针指向空;如 FILE *fp = NULL;
2、文件操作完成后,需要将文件关闭,一定要注意,否则会造成文件所占用内存泄漏和在下次访问文件时出现问题。
3、文件关闭后,需要将文件指针指向空,这样做会防止出现游离指针,而对整个工程造成不必要的麻烦;如:fp = NULL;