C语言 使用文件指针 操作硬盘下的文件

目录

一:C语言 文件

二:C语言 链表

三:fflush  刷新文件缓冲区

四:fread  文件读取操作

五:feof  读取文件的全部数据内容

六:ftell  读取文件的大小

七:目标文件 & 链表 综合编程应用


一: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) 按需分配

链表的学习:

  1. 链表的初始化
  2. 链表添加节点(尾部)
  3. 链表插入节点(pos)
  4. 链表删除节点(pos)
  5. 查询链表的节点信息(pos)
  6. 统计链表的节点个数
  7. 释放链表 (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;
}

文件操作的函数封装还有待完善,【读取文件数据信息内容、修改文件数据信息内容、追加文件数据信息内容  相关的函数封装以及接口测试】具体参考以下链接

文件链表综合应用

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

chenruhan_QAQ_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值