目录
一:C语言 文件
文件 程序运行在内存当中
数据永久保存 引入了文件 把数据存储到文件当中
打开文件 fopen(filename,”r+”);
r+ 打开已有的文件 不存在打开失败
w+ 文件不存在就创建 存在—创建—覆盖
FILE *f_open(filename);
读取文件 fread(读出的首地址,大小,单位数,文件指针)
写入文件 fwrite(写入数据的首地址,大小,单位数,文件指针)
文件指示器的移动:
fseek() //SEEK_END SEEK_SET
fseek(pf,0,SEEK_SET)开头
fseek(pf,0,SEEK_END) 末尾
保存文件 fflush(pf) stdin是标准输入输出文件
关闭文件 fclose()
二:C语言 链表
为什么要学习链表
1 解决数组的弊端 地址连续 内存闲置 不能扩容 插入删除涉及到大量的数据搬移
2 节点存储不连续 动态内存分配(malloc free) 按需分配
链表的学习:
- 链表的初始化
- 链表添加节点(尾部)
- 链表插入节点(pos)
- 链表删除节点(pos)
- 查询链表的节点信息(pos)
- 统计链表的节点个数
- 释放链表 (tmp->pdata) 要malloc开空间 (建议:用结构体指针定义)
三:fflush 刷新文件缓冲区
CFile.h
#ifndef CFILE_H
#define CFILE_H
#include<stdio.h> // 用到FILE结构体
//保证打开一个文件 char name[20] char name[ ] char* name
FILE *f_open(char *name);
#endif
CFile.c
#include"CFile.h"
//保证打开一个文件
FILE *f_open(char *name)
{
FILE *fp = NULL;
fp = fopen(name,"r+");
if(fp==NULL) //不存在
{
fp = fopen(name,"w+"); //就创建
}
}
main.c 接口测试
fflush文件缓冲区刷新数据信息 使用示例
#include<stdio.h>
#include<windows.h> //exit(0)
#include"CFile.h"
typedef struct menu_t
{
int id;
char name[20];
double price;
}MENU_T;
int main()
{
FILE *pf=NULL; //文件指针
MENU_T menu = {1001,"水煮鱼",59.8};
MENU_T menu1 = {1002,"水煮肉片",59.8};
MENU_T menu2 = {1003,"水煮田鸡",59.8};
MENU_T menu3 = {1004,"MILK",59.8};
// w+ 文件不存在创建文件 但要确认文件夹存在
pf = f_open("D:/menu.txt"); // 结果open success,文件所在地址:0042AAE8
if(pf==NULL)
{
printf("open fail\n");
exit(0);
}
else
{
printf("open success,文件所在地址:%p\n",pf);
}
//光标定位在文件开始 偏移量 +向右偏移 -向左偏移
fseek(pf,0,SEEK_END);
//1.写入内容的地址 2.写入的大小 3.几块 4.文件指针
fwrite(&menu3,sizeof(MENU_T),1,pf);
fflush(pf); //刷新文件缓冲区 保存进去
fclose(pf); //若是没有fflush,使用fclose也会保存
return 0;
}
fflush(pf);//刷新文件缓存区-保存进去
四:fread 文件读取操作
fread 使用示例
如下,文件指示器开头,读取文件中的一个数据信息 if语句
#include<stdio.h>
#include<windows.h> //exit(0)
#include"CFile.h"
typedef struct menu_t
{
int id;
char name[20];
double price;
}MENU_T;
int main()
{
FILE *pf=NULL; //文件指针
MENU_T menu = {1001,"水煮鱼",59.8};
MENU_T menu1 = {1002,"水煮肉片",59.8};
MENU_T menu2 = {1003,"水煮田鸡",59.8};
MENU_T menu3 = {1004,"MILK",59.8};
//创建结构体指针
MENU_T *pmenu = NULL;
pmenu = (MENU_T *)malloc(sizeof(MENU_T));
if(pmenu == NULL)
{
printf("open fail\n");
exit(0);
}
memset(pmenu,'\0',sizeof(MENU_T));
// w+ 文件不存在创建文件 但要确认文件夹存在
pf = f_open("D:/menu.txt"); // 结果open success,文件所在地址:0042AAE8
if(pf==NULL)
{
printf("open fail\n");
exit(0);
}
else
{
printf("open success,文件所在地址:%p\n",pf);
}
//文件指示器开头
rewind(pf);
if(fread(pmenu,sizeof(MENU_T),1,pf)>0)//可以拿文件里面数据
{
printf("%d\t%s\t%.1lf",pmenu->id,pmenu->name,pmenu->price);
}
fclose(pf);
return 0;
}
如下,文件指示器开头,读取文件的全部数据内容【将 if 改为 while】
//文件指示器开头
rewind(pf);
while(fread(pmenu,sizeof(MENU_T),1,pf)>0)
{
printf("%d\t%s\t%.1lf\n",pmenu->id,pmenu->name,pmenu->price);
}
fclose(pf);
五:feof 读取文件的全部数据内容
feof:可以读取文件中的全部数据内容,
使用示例如下
#include<stdio.h>
#include<windows.h> //exit(0)
#include"CFile.h"
typedef struct menu_t
{
int id;
char name[20];
double price;
}MENU_T;
int main()
{
FILE *pf=NULL; //文件指针
MENU_T menu = {1001,"水煮鱼",59.8};
MENU_T menu1 = {1002,"水煮肉片",59.8};
MENU_T menu2 = {1003,"水煮田鸡",59.8};
MENU_T menu3 = {1004,"MILK",59.8};
//创建结构体指针
MENU_T *pmenu = NULL;
pmenu = (MENU_T *)malloc(sizeof(MENU_T));
if(pmenu == NULL)
{
printf("open fail\n");
exit(0);
}
memset(pmenu,'\0',sizeof(MENU_T));
// w+ 文件不存在创建文件 但要确认文件夹存在
pf = f_open("D:/menu.txt"); // 结果open success,文件所在地址:0042BB08
if(pf==NULL)
{
printf("open fail\n");
exit(0);
}
else
{
printf("open success,文件所在地址:%p\n",pf);
}
//文件指示器开头
rewind(pf);
fread(pmenu,sizeof(MENU_T),1,pf);//读
while(feof(pf)==0) //判断文件是否读到结尾
{
printf("%d\t%s\t%.1lf\n",pmenu->id,pmenu->name,pmenu->price);
fread(pmenu,sizeof(MENU_T),1,pf);//再读
}
fclose(pf);
return 0;
}
结果同上,可以读取到文件中的全部数据内容
六:ftell 读取文件的大小
ftell:可以读取 文件有多大【文件数据字节数】,
使用示例如下
#include<stdio.h>
#include<windows.h> //exit(0)
#include"CFile.h"
typedef struct menu_t
{
int id;
char name[20];
double price;
}MENU_T;
int main()
{
FILE *pf=NULL; //文件指针
MENU_T menu = {1001,"水煮鱼",59.8};
MENU_T menu1 = {1002,"水煮肉片",59.8};
MENU_T menu2 = {1003,"水煮田鸡",59.8};
MENU_T menu3 = {1004,"MILK",59.8};
//创建结构体指针
MENU_T *pmenu = NULL;
pmenu = (MENU_T *)malloc(sizeof(MENU_T));
if(pmenu == NULL)
{
printf("open fail\n");
exit(0);
}
memset(pmenu,'\0',sizeof(MENU_T));
// w+ 文件不存在创建文件 但要确认文件夹存在
pf = f_open("D:/menu.txt"); // 结果open success,文件所在地址:0042BB08
if(pf==NULL)
{
printf("open fail\n");
exit(0);
}
else
{
printf("open success,文件所在地址:%p\n",pf);
}
//文件指示器开头
rewind(pf);
fread(pmenu,sizeof(MENU_T),1,pf);
while(feof(pf)==0)
{
printf("%d\t%s\t%.1lf\n",pmenu->id,pmenu->name,pmenu->price);
fread(pmenu,sizeof(MENU_T),1,pf);
}
printf("%d字节\n",ftell(pf)); //192字节
fclose(pf);
return 0;
}
七:目标文件 & 链表 综合编程应用
需要 CFile CList 以及 menu的 .h .c文件 从下面链接及源码中可自行领取学习
menu.h
#ifndef MENU_H
#define MENU_H
typedef struct menu_t
{
int menu_id;
char name[20];
double price;
struct menu_t *pnext;//指针 指向下个结构体存储的地址
}MENU_T;
//初始化链表头
MENU_T *menuList_init();
//把结构体添加到链表的尾部
void menuList_add(MENU_T *head,MENU_T menu);
//统计链表的节点个数
int menuList_count(MENU_T *head);
/*
获取某个节点的数据
参数1:链表头
参数2:第几个节点(不包括头)
返回值:找到节点的地址
1.遍历链表 指针域不为空计数
2.判断计数值与pos是否一样
如果一样 说明找到了 返回当前这个节点的地址
*/
MENU_T* getMenuInfoByPos(MENU_T *head,int pos);
/*
获取某个节点的数据
参数1:链表头
参数2:第几个节点(不包括头)
返回值:找到节点的地址
1.遍历链表 指针域不为空计数
2.菜品名称跟name一样
如果一样 说明找到了 返回当前这个节点的地址
*/
MENU_T* getMenuInfoByName(MENU_T *head,char *name);
/*
获取某个节点的数据
参数1:链表头
参数2:第几个节点(不包括头)
返回值:找到节点的地址
1.遍历链表 指针域不为空计数
2.菜品id跟传参的id一样
如果一样 说明找到了 返回当前这个节点的地址
*/
MENU_T* getMenuInfoById(MENU_T *head,int id);
//打印链表中节点的所有的数据--遍历
void MenuList_printInfo(MENU_T *head);
/*
在第几个节点的后面插入新节点,也就是菜品信息
1遍历整条链表,找到第几个节点
2找到之后插入
*/
void MenuList_InsertByPos(MENU_T *head,MENU_T menu,int pos);
/*
删除第几个节点
在第几个节点的后面删除节点,也就是菜品信息
1遍历整条链表,找到第几个节点
2找到之后删除
*/
int MenuList_DeleteByPos(MENU_T *head,int pos);
//释放整条链表
void freeMenuList(MENU_T *head);
#endif
menu.c
#include"menu.h"
#include<stdlib.h>
//初始化链表头
MENU_T *menuList_init()
{
//创建菜谱结构体指针变量
MENU_T *head = NULL;
//动态开空间
head = (MENU_T *)malloc(sizeof(MENU_T));
if(head == NULL)
{
printf("open fail\n");
exit(0);
}
//初始化 数据域
memset(head,'\0',sizeof(MENU_T));
//初始化指针域
head->pnext = NULL;
return head;
}
//把结构体添加到链表的尾部
void menuList_add(MENU_T *head,MENU_T menu)
{
MENU_T *tmp=head;
MENU_T *pNewNode = NULL; //新节点
//判断pnext是否为空 找到链表的尾部
while(tmp->pnext!=NULL)
{
tmp=tmp->pnext;
}
//动态开空间
pNewNode = (MENU_T *)malloc(sizeof(MENU_T));
if(pNewNode==NULL)
{
printf("open fail\n");
exit(0);
}
//初始化 数据域
memset(pNewNode,'\0',sizeof(MENU_T));
//数据域赋值
memcpy(pNewNode,&menu,sizeof(MENU_T));
//新节点连接到上一个节点的指针域
tmp->pnext = pNewNode;
}
//统计链表的节点个数
int menuList_count(MENU_T *head)
{
//每一个节点找pnext 判断一下是否为空
//不为空 +1
int count =0;
MENU_T *tmp=head;
while(tmp->pnext!=NULL)
{
count++;
tmp=tmp->pnext;
}
return count;
}
/*
获取某个节点的数据
参数1:链表头
参数2:第几个节点(不包括头)
返回值:找到节点的地址
1.遍历链表 指针域不为空计数
2.判断计数值与pos是否一样
如果一样 说明找到了 返回当前这个节点的地址
*/
MENU_T* getMenuInfoByPos(MENU_T *head,int pos)
{
//每一个节点找pnext 判断下一个是否为空
//不为空+1
int count =0;
MENU_T *tmp=head;
while(tmp->pnext!=NULL)
{
count++;
tmp=tmp->pnext;
if(pos==count) //是不是找到了
{
return tmp;
}
}
return NULL;
}
/*
获取某个节点的数据
参数1:链表头
参数2:第几个节点(不包括头)
返回值:找到节点的地址
1.遍历链表 指针域不为空计数
2.菜品名称跟name一样
如果一样 说明找到了 返回当前这个节点的地址
*/
MENU_T* getMenuInfoByName(MENU_T *head,char *name)
{
//每一个节点找pnext 判断下一个是否为空
//不为空+1
int count =0;
MENU_T *tmp=head;
while(tmp->pnext!=NULL)
{
count++;
tmp=tmp->pnext;
if(strcmp(name,tmp->name)==0) //是不是找到了
{
return tmp;
}
}
return NULL;
}
/*
获取某个节点的数据
参数1:链表头
参数2:第几个节点(不包括头)
返回值:找到节点的地址
1.遍历链表 指针域不为空计数
2.菜品id跟传参的id一样
如果一样 说明找到了 返回当前这个节点的地址
*/
MENU_T* getMenuInfoById(MENU_T *head,int id)
{
//每一个节点找pnext 判断下一个是否为空
//不为空+1
int count =0;
MENU_T *tmp=head;
while(tmp->pnext!=NULL)
{
count++;
tmp=tmp->pnext;
if(id==tmp->menu_id) //是不是找到了
{
return tmp;
}
}
return NULL;
}
//打印链表中节点的所有的数据--遍历
void MenuList_printInfo(MENU_T *head)
{
int count =0;
MENU_T *tmp=head;
while(tmp->pnext!=NULL)
{
tmp=tmp->pnext;
printf("%d\t%s\t%.1f\n",tmp->menu_id,tmp->name,tmp->price);
}
}
/*
在第几个节点的后面插入新节点,也就是菜品信息
1遍历整条链表,找到第几个节点
2找到之后插入
*/
void MenuList_InsertByPos(MENU_T *head,MENU_T menu,int pos)
{
//每一个节点找pnext判断一下是否为空
//不为空+1
int count =0;
MENU_T *tmp=head;
MENU_T *newNode =NULL;
while(tmp->pnext!=NULL)
{
count++;
tmp=tmp->pnext;
if(pos==count) //是不是找到了
{
//newNode开空间初始化,数据域内存copy
newNode=(MENU_T *)malloc(sizeof(MENU_T));
memset(newNode,'\0',sizeof(MENU_T));
memcpy(newNode,&menu,sizeof(MENU_T)); //数据域
//newNode pnext指向 当前的下一个
newNode->pnext=tmp->pnext;
//当前的下一个指向 新节点
tmp->pnext=newNode;
break;
}
}
}
//删除第几个节点
int MenuList_DeleteByPos(MENU_T *head,int pos)
{
int count =0;
MENU_T *tmp=head->pnext,*pre = head; //pre 记住上一个节点
while(tmp!=NULL)
{
count++;
if(count==pos)
{
//上个节点的pnext=tmp->pnext
pre->pnext=tmp->pnext;
free(tmp);
return 1;//删除成功
}
pre=pre->pnext;
tmp=tmp->pnext;
}
return -1;//没找到
}
//释放整条链表
void freeMenuList(MENU_T *head)
{
MENU_T *tmp=head;
while(tmp->pnext!=NULL)
{
head = head->pnext;
free(tmp);
tmp=head;
}
free(tmp);
}
创建菜谱 结构体指针,malloc动态开辟内存空间,memset初始化,对开辟并初始化后的内存空间操作,如写入数据至链表中,再将pmenu结构体指针数据信息,
通过fwrite(库自带函数)/File_write(自己封装的函数)写入至文件中,进而实现数据写入链表,再有链表写入[存储]文件
a.一次写入文件 一个数据信息 fwrite(库自带函数)
#include<stdio.h>
#include<windows.h> //exit(0)
#include"CFile.h"
#include"CList.h"
typedef struct menu_t
{
int id;
char name[20];
double price;
}MENU_T;
int main()
{
FILE *fp=NULL;
LIST_T *head = NULL;
//添加菜品 结构体指针
MENU_T *pmenu=NULL;
int count = 0;
//打开文件
fp = f_open("data/menu.txt");//自己创建data文件夹
if(fp==NULL)
{
printf("open file fail\n");
}
pmenu = (MENU_T *)malloc(sizeof(MENU_T));
if(pmenu==NULL)
{
printf("open fail\n");
}
memset(pmenu,'\0',sizeof(MENU_T));
head = List_init();
printf("请输入添加id:");
scanf("%d",&pmenu->id);
printf("请输入添加name:");
scanf("%s",pmenu->name);
printf("请输入添加价格:");
scanf("%lf",&pmenu->price);
// printf("%d\t%s\t%.1lf\n",pmenu->id,pmenu->name,pmenu->price);
List_add(head,pmenu);
count= List_count(head);
printf("节点个数%d\n",count);
fwrite(pmenu,sizeof(MENU_T),1,fp);
fflush(fp);
return 0;
}
b.一次写入文件 多个数据信息 [封装写入] File_write(自己封装的函数)
写入数据信息内容至文件中[函数封装] 调用接口测试如下
#include<stdio.h>
#include<windows.h> //exit(0)
#include"CFile.h"
#include"CList.h"
LIST_T *head = NULL;//全局变量
FILE *fp=NULL;//全局变量
typedef struct menu_t
{
int id;
char name[20];
double price;
}MENU_T;
void menuAdd()
{
MENU_T *pmenu=NULL;
int count = 0;
pmenu = (MENU_T *)malloc(sizeof(MENU_T));
if(pmenu==NULL)
{
printf("open fail\n");
exit(0);
}
memset(pmenu,'\0',sizeof(MENU_T));
printf("请输入添加id:");
scanf("%d",&pmenu->id);
printf("请输入添加name:");
scanf("%s",pmenu->name);
printf("请输入添加价格:");
scanf("%lf",&pmenu->price);
// printf("%d\t%s\t%.1lf\n",pmenu->id,pmenu->name,pmenu->price);
List_add(head,pmenu);
count= List_count(head);
printf("节点个数%d\n",count);
}
int main()
{
int i=0;
//打开文件
fp = f_open("data/menu.txt");
head = List_init();
if(fp==NULL)
{
printf("open file fail\n");
exit(0);
}
for(i=0;i<3;i++)
{
menuAdd();
}
File_write(fp,head,sizeof(MENU_T));
fclose(fp);
return 0;
}
文件操作的函数封装还有待完善,【读取文件数据信息内容、修改文件数据信息内容、追加文件数据信息内容 相关的函数封装以及接口测试】具体参考以下链接