-
C第十二天
今天讲的文件,内容有点多,但是都属于文件类的,到了后面把文件和链表结合的时候还是有点懵,自己写不出来。老师也说后期做项目主要由链表+结构体+文件构成,问题是现在这三样我一个都没掌握。老师给我们留了作业,倒是把这几天讲的内容都用上了。
文件:数据在硬盘上面的集合。是操作系统处理数据的单位。
文件格式分类:1.二进制文件 2.文本文件
二进制文件:存储的时候以二进制的形式把数据的补码进行存储。
=>
1111... 1111... 111... 111...
x * & %
49 50 51 52 53
11111 111.. 111.. 111.. 11...
1 2 3 4 5 ```
直接打开文件是以文本的方式显示,即一个字节一个字节的显示。
文本文件:存储的时候把每一个数据都看做是一个字符,存储他们的ASCii码。这样存储的时候一个字符占一个字节。直接打开文件,以文本的方式显示,即能看到原文。
文件分类:
从用户的角度:
特殊文件:标准输入输出文件或者设备文件。
标准输入:键盘 stdin
标准输出:显示屏 stdout
普通文件:普通磁盘文件
从操作系统的角度来看:每一个与主机相连的设备都是一个文件
文件类型结构体 FILE.
文件信息用系统定义的名为FILE的结构体描述,该结构体定义在stdio.h中,可以直接拿来用。
用法:文件打开时,系统自动建立文件结构体,并把指向它的指针返回出来,程序通过这个指针获得文件的信息,访问文件。文件关闭后,把文件结构体释放。
打开一个文件:
函数:```FILE *fopen(char *filename,char *mode);```
功能:按指定的方式打开文件
返回值:正常打开,为指向文件(结构体)的指针;失败,为NULL
>打开方式:
r:以只读的方式打开一个文本文件
rb:以只读的方式打开一个二进制文件
w:...只写..文本文件
wb:...只写..二进制
a:以追加的方式打开文本文件
ab:... 二进制
r+:以读写的方式打开文本文件
rb+:以读写的方式打开二进制文件
w+:以读写的方式打开文本文件
wb+:。。。。。。二进制文件
a+:以读写的方式打开文本文件
ab+:...........二进制文件
关闭一个文件:
函数:```int fclose(FILE *fp)```
功能:关闭fp指向的文件,释放文件结构体和文件指针
返回值: 成功,0;失败,非0
include <stdio.h>
include <stdlib.h>
void main()
{
FILE fp;
fp=fopen("3.c","a");
if(fp==NULL)
{
printf("open file is failed!\n");
exit(0);
}
int a=fclose(fp);
if(a==0)
{
printf("close is success!\n");
}
else
printf("close is failed!\n");
fp=NULL;
}
/
注:
1.若文件在当前路径下:
fp=fopen("文件名","mode");
2.若文件不在当前目录下:
fp=fopne("文件的绝对路径","mode");
3.若要手动输入要打开的文件名:
char filename[30];
scanf("%s",filename);
fp=fopen(filename,"r");
4.要对一个文件进行操作,必须先打开这个文件,操作完成之后,必须要关闭这个文件,不关闭的话容易造成数据丢失,关闭之后使fp=NULL;
5.以r,r+的方式打开文件,若文件不存在,则打开失败
以w,w+,a,a+的方式打开文件,若文件不存在,则会自动创建文件
6.以w/w+的方式打开文件,会把文件中原有的数据清除。
7.以a/a+的方式继续向文件中写入数据不会覆盖原有的数据。
*/
函数:```char fputc(char ch,FILE *fp)```
功能:把一个字符ch写到fp指向文件中
返回值:正确,返回ch,错误,返回EOF
>EOF:通常把它看做是一个特殊字符理解,其实在头文件stdio.h中它是常量-1的宏定义。在linux中,它并不是一个特殊字符,只是在系统读到文件结尾或者出错的时候所返回的一个信号值。
例:
#include <stdio.h>
void main()
{
char ch='a';
fputc(ch,stdout);
fputc('\n',stuout);
}
函数:```char fgetc(FILE *fp)```
功能:从fp指向的文件中读取一个字符
返回值:成功,返回读到的字符;当读到文件结尾或者出错的时候返回EOF
例:
include <stdio.h>
include <stdlib.h>
void main()
{
char ch=fgetc(stdin);
fputc(ch,stdout);
putchar('\n');
}
练习:把一个文件中的内容读到另一个文件。fgetc fputc
include <stdio.h>
include <stdlib.h>
void main()
{
//分别以读,写的方式打开2个文件2.c 3.c
FILE *fp1=fopen("2.c","r");
FILE *fp2=fopen("3.c","w");
if(fp1==NULL)
{
printf("open 2.c is failed!\n");
exit(0);
}
if(fp2==NULL)
{
printf("open 3.c is failed!\n");
exit(0);
}
char c;
//进行读写操作
while((c=fgetc(fp1))!=EOF){fputc(c,fp2);}
//关闭2个文件
int a=fclose(fp1);
int b=fclose(fp2);
if(a==0 && b==0)printf("文件关闭成功");
//指针置空
fp1=NULL;
fp2=NULL;
}
练习:从键盘输入一串字符,把它们存到磁盘文件中去,直到输入“#”结束。
方法1:
include <stdio.h>
include <stdlib.h>
void main()
{
//打开磁盘文件
FILE *fp;
if((fp=fopen("2.c","w"))==NULL)
printf("open is failed!\n");
//获取字符
char c;
while((c=getchar())!='#')
{
fputc(c,fp);
}
if(!fclose(fp))
printf("close is success!\n");
fp=NULL;
}
方法2:
include <stdio.h>
include <stdlib.h>
void main()
{
FILE *fp;
fp = fopen("4.c","w");
if(fp == NULL)
{
printf("打开失败!\n");
exit(0);
}
char c;
while((c = fgetc(stdin)) != '#')
{
fputc(c,fp);
}
int a = fclose(fp);
if(a == 0)
{
printf("关闭成功!\n");
}
fp = NULL;
}
函数:```long ftell(FILE *fp);```
功能:得知目前文件读写指针的位置
返回值:正确,返回目前文件读写指针的位置,就是当前文件读写指针所在的位置距离文件开头的长度。错误,返回-1.
例:
include <stdio.h>
include <stdlib.h>
void main()
{
FILE *fp=fopen("2.c","a");
long position=ftell(fp);
printf("%ld\n",position);
}
函数:```fscanf(FILE *fp,char *format,argc_list)```
功能:从fp指向的文件中按指定的格式获取数据的输入
返回值:成功,获取的数据的个数,失败,EOF
例:
char name[30];
scanf("%s%d",name,&age);
例:
scanf(fp,"%s%d",name,&age);
例:
include <stdio.h>
include <stdlib.h>
void main()
{
char name[30];
int age;
fscanf(stdin,"%s%d",name,&age);
printf("name:%s age:%d\n",name,age);
}
函数:```fprintf(FILE *fp,char *format,argc_list);```
功能:把数据按指定的格式输出到指定的文件中去。
返回值:成功,返回输出项的个数,失败,EOF
例:
include <stdio.h>
include <stdlib.h>
void main()
{
char name[30];
int age;
fscanf(stdin,"%s%d",name,&age);
fprintf(stdout,"name:%s age:%d\n",name,age);
}
练习:从键盘按指定的格式把数据输入到磁盘文件中去,然后又把磁盘文件中的数据读出来显示到屏幕上.fscanf/fprintf
include <stdio.h>
include <stdlib.h>
void main()
{
char name[30],name1[30];
int age,age1;
FILE *fp;
if((fp=fopen("3.c","w"))==NULL)
{
printf("open is failed!\n");
exit(0);
}
fscanf(stdin,"%s %d",name,&age);
fprintf(fp,"%s %d",name,age);
fclose(fp);
fopen("3.c","r");
fscanf(fp,"%s %d",name1,&age1);
fprintf(stdout,"%s %d",name1,age1);
fclose(fp);
}
函数:```void rewind(FILE *fp)```
功能:让文件指针重置在最开头。
include <stdio.h>
include <stdlib.h>
void main()
{
char name[30],name1[30];
int age,age1;
FILE *fp;
if((fp=fopen("3.c","w+"))==NULL)
{
printf("open is failed!\n");
exit(0);
}
fscanf(stdin,"%s %d",name,&age);
fprintf(fp,"%s %d",name,age);
long position=ftell(fp);
printf("%ld\n",position);
rewind(fp);//重置指针
position=ftell(fp);
printf("%ld\n",position);
fscanf(fp,"%s %d",name1,&age1);
fprintf(stdout,"%s %d",name1,age1);
fclose(fp);
}
函数:```int fseek(FILE *fp,long offset,int whence)```
功能:使fp的位置重置到指定的位置,从whence位置移动offset个字符
返回值:成功为0,失败为-1
whence的值:
0:文件开头
1:当前指针所指的位置
2:文件的结尾
例:
fseek(fp,100L,0);将文件指针fp移到离文件头100个字节的位置
fseek(fp,50L,1);将fp移到离当前位置后50个字节处
fseek(fp,-10L,2);将fp从文件尾倒移10个字节。
函数:```fread(addr,num数据块的大小,count数据块的个数,fp)```
功能:从文件中读取数据块.从文件fp中一次读取count个num大小的数据块到addr内存中。
返回值:成功,返回count;失败,返回0.
例:fread(&age,sizeof(int),1,fp);
函数:```fwrite(addr,int num,int count,FILE *fp);```
功能:往文件中写入数据块.从地址addr中,一次读取count个num大小的数据块到文件fp中。
例:fwrite(&age,sizeof(int),1,fp);
include <stdio.h>
include <stdlib.h>
void main()
{
int age=100,age1;
FILE *fp=fopen("x.c","w+");
if(fwrite(&age,sizeof(int),1,fp)==1)
//成功,返回一次读取的数据块的个数1,失败返回0
printf("写入成功!\n");
rewind(fp);//重置指针的位置为开头
if(fread(&age1,sizeof(int),1,fp)==1)
printf("读取成功!\n");
printf("%d\n",age1);
}
练习:从键盘输入4个学生的个人信息,把他们存到磁盘文件中去,然后读出来。
include <stdio.h>
include <stdlib.h>
struct student
{
char name[30];
int age;
}stu[4];
struct student stu1[4];
void main()
{
int i;
for(i=0;i<4;i++)
scanf("%s %d",stu[i].name,&stu[i].age);
FILE fp;
if((fp=fopen("student.txt","w+"))==NULL)
printf("open is failed!\n");
/
for(i=0;i<4;i++)
{
if(fwrite(&stu[i],sizeof(struct student),1,fp)!=1)
printf("写入失败!\n");
}/
/
for(i=0;i<4;i++)
{
if(fread(&stu1[i],sizeof(struct student),1,fp)!=1)
printf("读取失败!\n");
}*/
if(fwrite(stu,sizeof(struct student),4,fp)!=4)
printf("写入失败!\n");
rewind(fp);
if(fread(stu1,sizeof(struct student),4,fp)!=4)
printf("读取失败!\n");
for(i=0;i<4;i++)
printf("%s %d\n",stu1[i].name,stu1[i].age);
fclose(fp);fp=NULL;
}
homework:
新建一条链表,把链表里的数据读到文件中,然后把文件中的数据读到另一条链表中。