用C语言delphi编写任务管理器软件完成以下任务:已知一个文件file007.txt中有20个字符,请把这些数据读入到

用c语言编写的一个小程序,200行以上,急!!!_百度知道
用c语言编写的一个小程序,200行以上,急!!!
提问者采纳
这是一款图书管理系统 但是我没有写排序 没有写归还
因为归还和借出都差不多
里面有图书删除操作 学生查找 图书查找 学生输出 图书输出 图书借出
插入图书 插入学生 最多借三本书
没有对把数据格式化写入文件 而是块读入 读出
插入学生和图书 约定学生 图书看、号不为0
输入图书label
和 stu_num 为0时就结束插入 插入全部插在文件末尾
程序很简单 也是随便写的 不过我觉得接近我们这种初学者 以下附上代码#ifndef B_MANAGEMENT_H#define B_MANAGEMENT_H#include&stdio.h&#define MAX_BO 3#define LENTH_SB sizeof(sb_book)#define LENTH_ST sizeof(st_book)#define sb_book struct b#define st_book struct s//声明学生的数据结构struct s{ char name[20]; int
num_ char book[MAX_BO][30]; int
num_};//声明图书的数据结构struct
b{ long book_ char book_name[30]; int book_ int book_ sb_book*};//声明图书的操作sb_book* search_book(const char*,FILE*,sb_book*);//用线性存储结构存储文件数据void delete_book();void insert_book(sb_book*);//声明学生的操作st_book* search_student(int,FILE *p,st_book*);void insert_student(st_book*);void borrow_book(int,const char*,sb_book*,st_book*,FILE*,FILE*);#endifvoid main(){ delete_book();}#include&stdio.h&#include&stdlib.h&#include&bk_management.h&void insert_book(sb_book* pb_book){ FILE* fp=NULL; fp=fopen(&D:\\book.txt&,&a+&); if(fp==NULL) {
printf(&can not open file&);
exit(0); } if(fwrite(pb_book,LENTH_SB,1,fp)==0) {
printf(&failed to write string to book.txt&);
exit(0); }
fclose(fp);}#include&bk_management.h&#include&malloc.h&#include&stdlib.h&#include&stdio.h&void delete_book(){ long delete_ FILE* fp=NULL; sb_book *p1,*p2,*head=NULL; printf(&please input book_label:&); scanf(&%ld&,&delete_label); fp=fopen(&D:\\book.txt&,&r&); if(fp==NULL) {
printf(&can not open file&);
exit(0); } head=p1=p2=(sb_book*)malloc(LENTH_SB); fread(p1,LENTH_SB,1,fp); if(feof(fp)) exit(0);//将文件中的数据读入内存形成链表,head为链表首指针 while(!feof(fp)) {
p1=(sb_book*)malloc(LENTH_SB);
fread(p1,LENTH_SB,1,fp);
p2-&next=p1; } p1-&next=NULL; fclose(fp);//删除指定的书籍的数据块
while(p1-&book_label!=delete_label&&p1-&next!=NULL)
if(p1-&book_label==delete_label)
if(p1==head)
p2-&next=p1-&
printf(&no book you want to delete&);
//将数据块写会文件
fp=fopen(&D:\\book.txt&,&w&);
if(fp==NULL)
printf(&can not open file&);
while(head-&next!=NULL)
fwrite(head,LENTH_SB,1,fp);
head=head-&
fclose(fp);
}#include&stdio.h&#include&bk_management.h&void borrow_book(int num_student,const char* book_name,sb_book book[1000],st_book stu[1000],FILE* fp_sb,FILE* fp_st){ extern sum_ extern sum_ sb_book* book_ st_book* stu_ stu_st=search_student(num_student,fp_st,stu); if(stu_st-&num_book&MAX_BO) { book_sb=search_book(book_name,fp_sb,book); (book_sb-&book_remain)--; strcpy(stu_st-&book[stu_st-&num_book],book_name); (stu_st-&num_book)++; fseek(fp_sb,sum_book*LENTH_SB,0); fwrite(book_sb,LENTH_SB,1,fp_sb); fseek(fp_st,sum_student*LENTH_ST,0); fwrite(stu_st,LENTH_ST,1,fp_st); } else printf(&you can not borrow book!!!&);}#include&stdio.h&#include&stdlib.h&#include&bk_management.h&void main(){ FILE *fp_sb=NULL,*fp_st=NULL; int stu_ char _book[30]; sb_book book[1000]; st_book stu[1000]; fp_sb=fopen(&D:\\book.txt&,&r+&); fp_st=fopen(&D:\\stu.txt&,&r+&); if(fp_sb==NULL || fp_st==NULL) {
printf(&fail to open file&);
exit(0); } printf(&Please input student number:&); scanf(&%d&,&stu_num); getchar(); printf(&please input book:&); gets(_book); borrow_book(stu_num,_book,book,stu,fp_sb,fp_st); getch(); fclose(fp_sb); fclose(fp_st);}#include&stdio.h&#include&stdlib.h&#include&bk_management.h&void insert_student(st_book* pt_book){ FILE* fp=NULL; fp=fopen(&D:\\stu.txt&,&a+&); if(fp==NULL) {
printf(&can not open file&);
exit(0); } if(fwrite(pt_book,LENTH_ST,1,fp)==0) {
printf(&failed to write string to book.txt&);
exit(0); }
fclose(fp);}#include&stdio.h&#include&stdlib.h&#include&bk_management.h&void main(){ st_book stu[1000]; int i,j; FILE* fp=NULL; fp=fopen(&D:\\stu.txt&,&r&); if(fp==NULL) {
printf(&can not open file&);
exit(0); } for(i=0;i&1000;i++) {
fread(&stu[i],LENTH_ST,1,fp);
if(feof(fp))
printf(&%8d %-20s %8d\n&,stu[i].num_student,stu[i].name,stu[i].num_book);
for(j=0;j&stu[i].num_j++)
printf(&%-30s, &,stu[i].book[j]);
printf(&\n&); } getch(); fclose(fp);}#include&stdio.h&#include&bk_management.h&#include&stdlib.h&int sum_st_book* search_student(int num_stu,FILE* fp,st_book stu[1000]){
for(i=0;(!feof(fp))&&i&1000;i++)
fread(&stu[i],LENTH_ST,1,fp);
sum_student++;
} for(i=0;num_stu!=stu[i].num_student&&i&sum_i++) sum_student=i; if(num_stu==stu[i].num_student)
return &stu[i]; else
printf(&no student you search&); return NULL;}#include&stdio.h&#include&bk_management.h&#include&stdlib.h&void main(){ FILE* fp=NULL; st_book _stu[1000]; int num_ st_book* fp=fopen(&D:\\stu.txt&,&r&); if(fp==NULL) {
printf(&can not open file&);
exit(0); } printf(&please input student number:&); scanf(&%d&,&num_stu); stu=search_student(num_stu,fp,_stu); if(stu!=NULL) {
printf(&%8d %-20s %8d\n&,stu-&num_student,stu-&name,stu-&num_book);
for(j=0;j&stu-&num_j++)
printf(&%-30s, &,stu-&book[j]);
printf(&\n&); } fclose(fp); getch();}#include&stdio.h&#include&stdlib.h&#include&bk_management.h&void main(){ sb_book* FILE* fp=NULL; fp=fopen(&D:\\book.txt&,&r&); if(fp==NULL) {
printf(&can not open file&);
exit(0); } while(1) {
p=(sb_book*)malloc(LENTH_SB);
fread(p,LENTH_SB,1,fp);
if(feof(fp))
printf(&%8d %-30s %-3d %-3d\n&,p-&book_label,p-&book_name,p-&book_num,p-&book_remain); } fclose(fp); getch();
}#include &bk_management.h&#include&malloc.h&void main(){ sb_book * while(1) { pb=(sb_book*)malloc(LENTH_SB); printf(&please input book_label:&); scanf(&%ld&,&pb-&book_label); getchar(); printf(&please input book_name:&); gets(pb-&book_name); printf(&please input book_num:&); scanf(&%d&,&pb-&book_num); pb-&book_remain=pb-&book_ if(pb-&book_label==0) insert_book(pb); } }#include &bk_management.h&#include&malloc.h&void main(){ st_book* while(1) { pt=(st_book*)malloc(LENTH_ST); printf(&please input num_student:&); scanf(&%d&,&pt-&num_student); getchar(); printf(&please input name:&); gets(pt-&name); strcpy(pt-&book[0],&0&); strcpy(pt-&book[1],&0&); strcpy(pt-&book[2],&0&); pt-&num_book=0; if(pt-&num_student==0) insert_student(pt); } }#include&stdio.h&#include&bk_management.h&#include&stdlib.h&int sum_sb_book* search_book(const char* ptr,FILE* fp,sb_book book[1000]){
for(i=0;(!feof(fp))&&i&1000;i++)
fread(&book[i],LENTH_SB,1,fp);
sum_book++;
} for(i=0;strcmp(&book[i].book_name,ptr)&&i&sum_i++) sum_book=i; if(!strcmp(&book[i].book_name,ptr))
return &book[i]; else
printf(&no book you search&); return NULL;}#include&stdio.h&#include&stdlib.h&#include&bk_management.h&void main(){ FILE* fp=NULL; char book_name[30]; sb_book* sb_book _book[1000]; fp=fopen(&D:\\book.txt&,&r&); if(fp==NULL) {
printf(&can not open file&);
exit(0); } printf(&please input book_name:&); gets(book_name); book=search_book(book_name,fp,_book); if(book!=NULL) printf(&%8d %-30s %-3d %-3d\n&,book-&book_label,book-&book_name,book-&book_num,book-&book_remain); fclose(fp); getch();}
提问者评价
其他类似问题
1人觉得有用
为您推荐:
其他1条回答
C语言课程设计报告-------学生成绩简单管理程序一、系统菜单的主要功能(1)输入若干条记录(2)显示所有记录(3)按学号排序(4)插入一条记录(5)按姓名查找,删除一条记录(6)查找并显示一条记录(7)输出统计信息 (新增)(8)从正文中添加数据到结构体数组中(9)将所有数据写入文件中(0)退出程序二、题目分析该题主要考察学生对结构体,指针,文件的操作,以及C语言算法的掌握,所以完成此道题目要求较强的设计能力,尤其是要有一种大局观的意识。如何调程序也非常重要,通过这个程序可以学习到以前调试短程序没有的的经验。菜单中的每一个选项都对应一个子程序,子程序的算法几乎囊获了所有C语言学过的技巧,下面就各个子程序中的功能进行说明:功能1和4的算法相似,输入一条记录到结构体中去,其中...
小程序的相关知识
等待您来回答
下载知道APP
随时随地咨询
出门在外也不愁利用c语言编写
读出一文件名为file的内容并显示_百度知道
利用c语言编写
读出一文件名为file的内容并显示
提问者采纳
本例程序的功能是从文件中逐个读取字符,在屏幕上显示。 程序定义了文件指针fp,以读文本文件方式打开文件“file.txt”, 并使fp指向该文件。如打开文件出错, 给出提示并退出程序。程序第12行先读出一个字符,然后进入循环, 只要读出的字符不是文件结束标志(每个文件末有一结束标志EOF)就把该字符显示在屏幕上,再读入下一字符。每读一次,文件内部的位置指针向后移动一个字符,文件结束时,该指针指向EOF。执行本程序将显示整个文件。#include&stdio.h&main(){FILE *if((fp=fopen(&file.txt&,&rt&))==NULL){printf(&Cannot open file strike any key exit!&);getch();exit(1);}ch=fgetc(fp);while (ch!=EOF){putchar(ch);ch=fgetc(fp);}fclose(fp);}
其他类似问题
为您推荐:
c语言的相关知识
其他1条回答
不用我做了,看这个就可以了。
标准文件函数
标准文件函数主要包括文件的打开、关闭、读和写等函数。不象BASIC 、
FORTRAN语方有顺序文件和随机文件之分,
在打开时就应按不同的方式确定。
Turbo C2.0并不区分这两种文件, 但提供了两组函数, 即顺序读写函数和随机读
一、文件的打开和关闭
任何一个文件在使用之前和使用之后, 必须要进行打开和关闭, 这是因为操
作系统对于同时打开的文件数目是有限制的, DOS操作系统中,
可以在DEVICE
.SYS中定义允许同时打开的文件数n(用files=n定义)。其中n 为可同时打开的文
件数, 一般n&=20。因此在使用文件前应打开文件, 才可对其中的信息进行存取。
用完之后需要关闭, 否则将会出现一些意想不到的错误。Turbo C2....
等待您来回答
下载知道APP
随时随地咨询
出门在外也不愁求解:用C语言编写程序,从键盘上输入100个字符,存入名为“file.txt“文件中。。_百度知道
求解:用C语言编写程序,从键盘上输入100个字符,存入名为“file.txt“文件中。。
提问者采纳
#include &stdio.h& main() { FILE *p=fopen(&file.txt&,&w&); char s[128]; scanf(&%s&,s); // 读取多少都可以,但不要超过127 fprintf(p,&%s\n&,s); // 存入到&file.txt”文件 }
其他类似问题
为您推荐:
等待您来回答
下载知道APP
随时随地咨询
出门在外也不愁用C语言编写软件完成以下任务:已知一个文件file007.txt中有20个字符,请把这些数据读入到_百度知道
用C语言编写软件完成以下任务:已知一个文件file007.txt中有20个字符,请把这些数据读入到
用C语言编写软件完成以下任务:已知一个文件file007.txt中有20个字符,请把这些数据读入到数组中,按照由小到大的顺序排列后,再写入到另一个文件myfile.dat中。用户界面中的菜单至少应包括“显示file007数据”,“生成myfile.dat文件”,“显示myfile.dat数据”,“退出”四项。
提问者采纳
#include&&stdio.h&#include&&stdlib.h&void&display(char*&fname)&{ int&i; FILE&* if((fp&=&fopen(fname,&&r&))&==&NULL)&{
printf(&File&open&error!&);
exit(0); } for(i=0;&i&20;&i++)
putchar(fgetc(fp)); putchar('\n'); fclose(fp);}void&generate(char*&infname,&char*&outfname)&{ FILE&*fip,&* int&i; if((fip&=&fopen(infname,&&r&))&==&NULL)&{
printf(&File&open&error!&);
exit(0); } if((fop&=&fopen(outfname,&&w&))&==&NULL)&{
printf(&File&open&error!&);
exit(0); } for(i=0;&i&20;&i++)
fputc(fgetc(fip),&fop); fclose(fip); fclose(fop);}int&main(){ int&choice&=&0; do&{
printf(&Please&select&your&action:\n&);
printf(&1.&Display&file007&data.\n&);
printf(&2.&Generate&myfile&data.\n&);
printf(&3.&Display&myfile&data.\n&);
printf(&4.&Exit.\n&);
scanf(&%d&,&&choice);
switch(choice)&{
display(&file007.txt&);
generate(&file007.txt&,&&myfile.dat&);
display(&myfile.dat&);
} }&while(choice&!=&4); return&0;}
提问者评价
太给力了,你的回答完美的解决了我的问题!
来自团队:
其他类似问题
为您推荐:
等待您来回答
下载知道APP
随时随地咨询
出门在外也不愁所谓文件(file)一般指存储在外部介质上数据的集合,比如我们经常使用的mp3、mp4、txt、bmp、jpg、exe、rmvb等等。这些文件各有各的用途,我们通常将它们存放在磁盘或者可移动盘等介质中。那么,为什么这里面又有这么多种格式的文件呢?原因很简单,它们各有各的用途,区分就在于这些文件里面存放的数据集合所遵循的存储规则不一样。举个例子比如bmp图片文件,为什么他能够表示一张图片,因为它有固定的格式,哪一段到哪一段,哪个偏移到哪个偏移应该存放什么数据是规定好了的。比如有文件头,一般是一个结构体,存放的文件的一些信息,如图片的大小,像素等等。再后来有数据区。然后我们要显示一张图片,就只需要按照前面所说的规则将文件头结构和数据块读出来,然后将这些数据在屏幕上用颜色表示出来,就成了一张图片。其它文件格式也类似。
这里要说一个更重要的例子,对我们理解文件有好处。那么这个文件就是exe文件(这里只讨论windows平台),通常我们认为它是一个可执行程序,这无疑是增加了它的神秘度。从本质上来讲exe无非是一种固定的文件格式罢了。既然这样,它就有一套自己的存储规则。跟前面的图片文件一样有规则。此时,你可能会问:你这么说那我就可以纯手工(直接填写数据填充文件)写出一个exe可执行文件了? 面对你这个问题,我只能说你已经习惯思考了,已经习惯给自己提问了,已经很聪明了。那么答案是肯定的,你完全可以用一个编辑器直接填写数据写出一个helloworld.exe文件或者helloworld.dll文件。因为这些具有一定格式规则的文件一般是二进制存储的,于是我们可以用一个二进制编辑器新建一个二进制文件,然后向里面填写数据。然后双击运行输出&helloworld&字符串。你可能会觉得很有成就感,我之前就写过一个exe和dll。这里exe和dll的文件格式也就是著名的PE文件格式。有兴趣你可以去查阅相关资料,此非本文重点。
总结上面的认识,文件无非就是一段数据的集合,这些数据可以是有规则的集合,也可以是无序的集合。操作系统也就是以文件为单位对数据进行管理的。也就是说,要访问外部介质上的数据,必须先按照文件名进行查找,然后从该文件中读取数据。要想写数据到外部介质,必须得建立一个文件,然后再写入。因此,这样来看,你眼前的文件将是一堆一堆数据而已,也没有什么类型文件之分了,类型只是为了区分而已,假如你把一个exe文件的扩展名改为txt,把它用记事本打开,同样是可行的,只是会执行exe文件里面的东西而已。(这里又不得不提到一点,如果你是一名程序员或者爱好者,那么你不应该将你的文件扩展名给隐藏了,要让它显示出来,如果你隐藏了,无非是增加了它的神秘感,同时在文件操作上不方便。通过上面的本质,我相信你能体会到我为什么这么说。)
说到这里,你应该知道文件是什么了,那么再来看二进制文件和ASCII文本文件,为什么要分为这两种呢?
首先、文本文件方式存储多用于我们需要明显知道文件里面的内容时,比如ini、h、c等文件都是文本文件,这种文件存储的是字符(ASCII码),比如一个整数10000,类型是short,占2字节,存储文本形式将占用5个字节,一共5个字符。你可以想想更多的例子,体会文本文件方便之处(提示:这里的文本文件不是说是txt文件,而是指所有以文本格式存储的文件。)
其次、二进制文件方式多用于直接将内存里面的数据形式原封不动存放到文件里,比如上面的short 10000,在内存中占2字节,存储内容为10000的二进制数,存到文件后还是占2字节,内容也是10000的二进制。这种方式可以整块数据一块儿存储,同时还可以将内存数据映射到文件里。
由上面两点,C语言操作文件可以是字节流或者二进制流。它把数据看成是一连串字符(字节),而不需要考虑边界。C语言对文件的存取是以字节为单位的。输入输出的数据流的开始和结束仅受程序控制而不受物理符号(如回车换行符)控制。这种文件通常称为流式文件,大大增加了灵活性。我们可以产生很多自己的文件格式,在游戏程序里面,用得比较多的就是资源包的格式,一般就是自定义的存取规则。我之前也写了一个包文件,存取只需要遵循规则,原理是非常简单的。大家可以试试在脑子里面构造一个包文件。
在ANSI C标准中,使用的是&缓冲文件系统&。所谓缓冲文件系统指系统自动地在内存区为每一个正在使用的文件名开辟一个缓冲区,从内存向磁盘输出数据必须先送到内存中的缓冲区,装满后再一起送到磁盘去。反向也是如此。这里需要说明两个词:&输入&&输出&。输入表示从文件里读数据到程序里,输出表示从程序里写数据到文件中。
了解了文件及文件存储形式,下面该正式进入文件的读写了,不要太激动,还是慢慢来。细节往往决定成败。在缓冲文件系统中,有一个很重要的一个东西就是文件指针,每个使用的文件都会在内存中开辟一个区,用于存放文件的有关信息,这些文件信息就保存在一个结构体变量中的,这个结构体是由系统定义的,名为FILE,先来看看VC2005在stdio.h下FILE结构体的定义:
struct _iobuf
// 指向buffer中第一个未读的字节
// 记录剩余未读字节的个数
// 指向一个字符数组,即这个文件的缓冲
// FILE结构所代表的打开文件的一些属性
// 用于获取文件描述,可以使用fileno函数获得此文件的句柄。
// 单字节的缓冲,即缓冲大小仅为1个字节,如果为单字节缓冲,_base将无效
// 记录这个缓冲的大小
// temporary file (i.e., one created by tmpfile()
// call). delete, if necessary (don't have to on
// Windows NT because it was done by the system when
// the handle was closed). also, free up the heap
// block holding the pathname.
typedef struct _iobuf FILE;
好了,上面的结构体就是这样定义的。这里不得不再次提到缓冲:
常量(mode)
无缓冲模式
该文件不使用任何缓冲,也可以说是字节缓冲
只能保存一个字节。
行缓冲模式
仅对文本模式打开的文件有效,所谓行,即是指每收到一个换行符(/n或/r/n),就将缓冲flush掉
全缓冲模式
仅当缓冲满时才进行flush
上面结构体中的_flag就标记了缓冲的信息(我们关心这三个):
#define _IOYOURBUF
// 使用用户通过setbuf提供的buffer
#define _IOMYBUF
// 这个文件使用内部的缓冲
#define _IONBF
// 无缓冲模式
#define _IOLBF
// 行缓冲模式
#define _IOFBF
// 全缓冲模式
同时,_flag也标记了读写模式,比如"r+"、"w+"等。
#define _IOREAD
#define _IOWRT
#define _IORW
// 可读可写
上面的3中模式就是"r"、"w"、"+"任意组合起来表示的意思。
正因为使用缓冲模式,是为了避免频繁的系统调用开销,有了缓冲就不需要每次都访问实际的文件。当然缓冲也会带来隐患,比如写文件时,先是到缓冲,如果此时系统崩溃或者进程意外退出时,有可能导致文件数据的丢失。因此C语言提供了几个基本的函数,弥补缓冲带来的问题:
int fflush( FILE* stream )
// flush指定文件的缓冲,若参数为NULL,则flush所有文件的缓冲。
int setvbuf( FILE *stream, char* buf,
int mode, size_t size )
// 设定缓冲类型,如上面的表格。
void setbuf( FILE* stream,
char* buf )
// 设置文件的缓冲,等价于( void )setvbuf( stream, buf, _IOFBF, BUFSIZ ).
所谓flush一个缓冲,是指对写缓冲而言,将缓冲内的数据全部写入实际的文件,并将缓冲清空,这样可以保证文件处于最新的状态。之所以需要flush,是因为写缓冲使得文件处于一种不同步的状态,逻辑上一些数据已经写入了文件,但实际上这些数据仍然在缓冲中,如果此时程序意外地退出(发生异常或断电等),那么缓冲里的数据将没有机会写入文件。flush可以在一定程度上避免这样的情况发生。
在这个表中我们还能看到C语言支持两种缓冲,即行缓冲(Line Buffer)和全缓冲(Full Buffer)。全缓冲是经典的缓冲形式,除了用户手动调用fflush外,仅当缓冲满的时候,缓冲才会被自动flush掉。而行缓冲则比较特殊,这种缓冲仅用于文本文件,在输入输出遇到一个换行符时,缓冲就会被自动flush,因此叫行缓冲。
终于把概念性的东西和准备步骤做完了,下面该看看具体的读写文件了。有了前面的准备工作,读写文件将不是难事了,因为有现成的库函数供我们使用,我们下面的段落将是如何使用这些库函数和一些注意事项而已了。
首先看如何打开文件,先看代码:
#include &stdio.h&
int main( void )
FILE* pReadFile = fopen( "E://mytest.txt", "r" );
// 打开文件
if ( pReadFile == NULL )
fclose( pReadFile );
// 关闭文件
上面的这段代码,只是一个简单的打开文件,如果成功打开后直接关闭。这里打开的是一文本文件,是以只读的方式打开。使用fopen函数打开,第一个参数是文件路径,第二个参数是读写模式,返回值为0表示打开失败。先看看读写模式:
&文件使用方式
"r"(只读)&
为输入打开一个文本文件,不存在则失败
为输出打开一个文本文件,不存在则新建,存在则删除后再新建
&"a"(追加)
向文本文件尾部增加数据,不存在则创建,存在则追加
'rb"(只读)&
为输入打开一个二进制文件,不存在则失败
"wb"(只写)&
为输入打开一个二进制文件,不存在则新建,存在则删除后新建
"ab"(追加)&
向二进制文件尾部增加数据,不存在则创建,存在则追加
"r+"(读写)&
为读写打开一个文本文件,不存在则失败
"w+"&(读写)
为读写建立一个新的文本文件,不存在则新建,存在则删除后新建
&"a+"(读写)
为读写打开一个文本文件,不存在则创建,存在则追加
"rb+"(读写)
为读写打开一个二进制文件,不存在则失败
"wb+"(读写)
为读写建立一个新的二进制文件,不存在则新建,存在则删除后新建
&"ab+"(读写)
为读写打开一个二进制文件,不存在则创建,存在则追加
一、读写字符
C语言为从文件中读写一个字符提供了两个函数:
int __cdecl fgetc( FILE* stream );
// 从文件读入一个字符
int __cdecl fputc( int ch, FILE* stream );
// 写入一个字符到文件
#include &stdio.h&
int main( void )
FILE* pReadFile = fopen( "E://mytest.txt", "r" );
// 打开文件
if ( pReadFile == NULL )
while ( ( cInput = fgetc( pReadFile ) ) != EOF )
// 从文件读入一个字符,如果到文件尾部,则返回EOF(-1)
printf( "%c", cInput );
fclose( pReadFile );
// 关闭文件
假如mytest.txt文件的内容是:
三行,那么我们逐个读入每个字符,直到EOF结束,EOF很简单,其实就是#define EOF (-1),WINDOWS为了能够返回失败为-1,因此fgetc的返回值使用是int类型。同时-1也不是某个字符的ASCII,所以不影响,一举两得。上面程序while循环不断从文件中读取单个字符,遇到换行符(WINDOWS下回车符('/r')为13, 换行符('/n')为10),printf输出后变处理成换行符了,因此文件里面3行,逐个读入程序里在终端显示后还是3行。代码很简单,就不用多说了。这里需要提到一点:
问题一:当第一次执行了fgetc后,我们看看pReadFile指针里面的内容与刚执行了fopen函数后的内容有所变化,为什么?
再来看fputc函数:
#include &stdio.h&
int main( void )
int i = 0;
char szOutput[ 32 ] = "masefee/nhello";
FILE* pWriteFile = fopen( "E://mytest.txt", "w" );
// 打开文件
if ( pWriteFile == NULL )
while ( szOutput[ i ] != 0 )
fputc( szOutput[ i ], pWriteFile );
// 写入一个字符到文件
fclose( pWriteFile );
// 关闭文件
我特意在szOutput数组里写了一个'/n'字符,此字符就是换行符newline,意图是当输出到e之后,便输出一个换行符,让字符串换行。因此最终mytest.txt文件里面的内容如下:
到这里,你可能会想到第一个fgetc的例子是我们预先在文件中输入3行字符,然后读入到程序中。我们在用记事本输入3行文本的时候,每当换行的时候我们敲键盘是按的回车。
问题二:既然我们敲的是回车,为什么在文件里存储的是'/n'而不是'/r'?
同时,到这里想到第一个问题,我们又来观察一下,当刚使用fopen函数时,pWriteFile里面的内容是:
pWriteFile&&&&&&&&&
0x00437bb0
_ptr&&&&&&&&&&&&&&&&&&&0x
_cnt&&&&&&&&&&&&&&&&&&
_base&&&&&&&&&&&&&&&
_flag&&&&&&&&&&&&&&&&&
_file&&&&&&&&&&&&&&&&&&&3
_charbuf&&&&&&&&&&&&0&
_bufsiz&&&&&&&&&&&&&&0&
_tmpfname&&&&&&
而执行了fputs函数,到换行符后我们再看pWriteFile里面的内容:
pWriteFile&&&&&&&&&
0x00437bb0
_ptr&&&&&&&&&&&&&&&&&&&0x
_cnt&&&&&&&&&&&&&&&&&&
_base&&&&&&&&&&&&&&&
_flag&&&&&&&&&&&&&&&&&
_file&&&&&&&&&&&&&&&&&&&3
_charbuf&&&&&&&&&&&&0&
_bufsiz&&&&&&&&&&&&&&4096
_tmpfname&&&&&&
然后我们再看看_base所在内存的值:
6d 61 73 65 66 65 65&0a&68
&m& a&& s&&
e&&& f&& e&&&
从这个现象我们能够意识到,FILE结构里面_base所指向的缓冲区,_cnt表示还剩下多少个字节没有写。还可以意识到,我们在不设置任何参数时,默认情况下是采用的全缓冲模式,填充4096字节后自动会写入到文件,在这里我们没有那么多字节,因此在fclose函数执行后,文件里便写入了值。你可以打断点在fclose上,等程序断下来后,观察你磁盘里面的mytest.txt是空的,当执行了fclose后大小就变了。这也能体现缓冲区的一个现象。
同样,如果你想立即将缓冲区的数据写到文件里,可以在fclose函数前面加上:
fflush( pWriteFile );
当执行完此函数后,数据便写进了文件,最后再关闭文件。
二、读写字符串
C语言为从文件中读写字符串提供了2个函数:
char* __cdecl fgets( char* _Buf, int _MaxCount, FILE* _File );
参数一:要从文件中读入字符串的存放空间。
参数二:最大读取字节数。
参数三:文件指针。
返回值:返回读入的字符串指针。
__cdecl fputs( const char* _Str,
FILE* _File );
参数一:要写入文件的字符串
参数二:文件指针
返回值:失败或成功,0表示成功,其它表示失败。
先来看字符串读取:
#include &stdio.h&
int main( void )
szInput[ 32 ] = { 0 };
char* pRet = NULL;
FILE* pReadFile = fopen( "E://mytest.txt", "r" );
// 打开文件
if ( pReadFile == NULL )
pRet = fgets( szInput, 32, pReadFile );
// 从文件中读取一个字符串到szInput数组中
fclose( pReadFile );
// 关闭文件
其它函数不说了,这里只说fgets函数,第二个参数传的是32,实际只能从文件中读取31个字符,因为fgets函数内部会将最后一个字符置为'/0', 表示字符串结束。那么我们可以看看fgets函数的内部原理,我这里写写伪代码,为了更清晰的表现出来:
fgets( char* dst, int maxcount, FILE* file )
while( --maxcount )
ch = readFromFile();
if ( ( *dst++ =
ch ) == '/n' )
// 赋值为'/0'
如果最大读取字节数量足以读到换行,将停止读取字符,然后阶数本字符串,然后返回。
明白了fgets函数,fputs函数就简单了:
#include &stdio.h&
int main( void )
char szOutput[ 32 ] = "masefee/nhello";
FILE* pWriteFile = fopen( "E://mytest.txt", "w" );
// 打开文件
if ( pWriteFile == NULL )
fputs( szOutput, pWriteFile );
// 写入一个字符串到文件
fclose( pWriteFile );
// 关闭文件
这里我也专门为字符数组里增加了一个换行符,写入字符串的时候并不会因为换行符而只写换行符前面的字符,同时在fputs内部会求第一个参数的长度strlen( Str&); 然后再写入这么一个长度的字符串到文件。
到这里又得提醒一点,即便是文件里面含有'/0'(ASCII码为0的字符)。fgets函数同样会一直读取到换行符或者读取规定的字符个数(此字符个数小于一行字符数)。虽然是读了一行,中间因为有0,因此字符串被截断,读出来的字符串并没有一行,只有0前面的所有字符。这里大家需要注意。同时fputs函数会以0结束写入文件,这是跟通常情况一样的,可以不用关心。
三、格式化数据读写
C语言既然有printf、scanf,那么同样也有文件操作的格式化函数:
int __cdecl fprintf( FILE* _File, const char* _Format, ... );
int __cdecl fscanf( FILE* _File, const char* _Format, ... );
这两个函数跟printf和scanf的用法非常相似,只是这里输入输出是关于文件的。
直接贴代码:
#include &stdio.h&
typedef struct SStudent
name[ 11 ];
int main( void )
FILE* pReadFile = fopen( "E://mytest.txt", "r" );
// 打开文件
if ( pReadFile == NULL )
fscanf( pReadFile, "%d%s", &stu.number, &stu.name );
fclose( pReadFile );
// 关闭文件
我定义了一个结构体,里面一个学号,一个姓名。然后打开文件,读取数据到stu结构体变量中。假如文件中是:
345&& masefee
然后读到stu结构体变量中,number为345,name为"masefee"。
fscanf读取数据是以空格、制表符、换行符进行分割的,我们可以这样来填充结构体。
再来看fprintf:
#include &stdio.h&
typedef struct SStudent
name[ 11 ];
int main( void )
FILE* pWriteFile = fopen( "E://mytest.txt", "w" );
// 打开文件
if ( pWriteFile == NULL )
stu.number = 100;
strcpy( stu.name, "masefee" );
fprintf( pWriteFile, "%d
%s", stu.number, stu.name );
fclose( pWriteFile );
// 关闭文件
此程序将把结构体stu的内容写到文件里,注意这里的name不会把结束符'/0'写到文件里。
好了,说到这里,上面几个基本的文件操作函数已经写完了,我只是使用了"r"和"w"两种方式,其它方式你可以自行测试,也没有什么特别的。如果你是用上面的函数去读取二进制序列,也是没有错的,只不过你更不好控制而已。至于和"+"组合也没有什么特别的,无非就是在文件尾部追加,原理一样,大家可以自行测试。
四、文件数据块读写
同样C语言也提供了两个函数:
size_t __cdecl fwrite
  const void *buffer,
// 要写入文件的数据块
  size_t size,
// 写入文件的字节数
  size_t count,
// 写入count个size大小的数据
  FILE *stream
// 文件指针
size_t __cdecl fread
  void * _DstBuf,
// 存放从文件读出来的数据
  size_t _ElementSize,
// 读取字节数
  size_t _Count,
// 读入次数
  FILE * _File
// 文件指针
先看看fwrite函数:
#include &stdio.h&
typedef struct SStudent
name[ 12 ];
int main( void )
FILE* pWriteFile = fopen( "E://mytest.txt", "w" );
// 打开文件
if ( pWriteFile == NULL )
stu.number = 10000;
strcpy( stu.name, "masefee" );
fwrite( &stu, sizeof( stu ), 1, pWriteFile );
fclose( pWriteFile );
// 关闭文件
这样写入文件后,mytest.txt的内容为:
'& masefee 烫烫
你可能会疑惑,为什么会有乱码?而且还有可恶的&烫&字。原因很简单,fwrite函数是以数据块的形式写数据到文件的,比如这里的stu结构体变量,我们将它整块写入文件,一共16字节,因此上面的乱码对应的就是stu结构体变量在内存中的存放形式,number占4字节,name占12字节,具体的数值是:
10 27 00 00&6d 61 73 65 66 65 65 00&cc cc cc cc
&&&&10000&&&&&&&&&&&&&&&&&"masefee"&&&&&&&&&&&&&&
因为我们在为name拷贝字符串时,并没有将name的所有字符清零,因此系统默认初识化为0xcc,为什么初始化为0xcc,之前我应该提过,主要是这个0xcc是汇编中断指令的机器码,主要防止访问越解释,进行中断报错。而0xcccc就是中文编码的&烫&字。
最后面的两个&烫&还不能省略,因为我们是以块写入文件的,如果去掉4个cc,那么将没有16字节,如果有多个结构体变量的数据一块儿写到文件中时,结构体的数据对齐是非常重要的,否则将读写越界,跟内存一样。这里就好比内存的一个映射。
至于为什么会出现乱码,是因为超过可现实ASCII码值,看上去就是乱的,其实数据还是正常的。
理解了fwrite函数后,fread函数就简单了,由于篇幅原因我这里只写关键:
Student stu_
fread( &stu_out, sizeof( Student ), 1, pReadFile );
这样就能填充好stu_out结构体变量,我想你已经体会到了数据块读写时,数据对齐的重要性了。在游戏的资源包,就是采用的数据块的存储形式,同时bmp、jpg、exe、dll等文件都是由很多个数据块,通常是结构体的形式直接写入文件的,这样文件头记录了很多偏移,很多大小等就显得非常重要了。
最后,我直接写了一个实例,就是简单的打包,解包程序。可以将多个文件放置到一个包文件里,这个包是二进制包。基本的功能已经实现,只需要添加比如压缩,界面等优化工作了。我初步测试了一下是可以成功打包解包的,也没有太多的条件检查和效率考虑,本文重在解释文件操作的灵活性和重要性。
#include &stdio.h&
#include &string.h&
#include &stdlib.h&
typedef unsigned int
typedef unsigned char byte;
// 包文件中最大可容纳的文件个数
#define MAX_FILE_COUNT 10
// 全局包文件指针
g_pMasFile = NULL;
// 资源包文件头结构
typedef struct SMaseFileHeader
// 包文件头标记: 'MASE'
// 包内文件个数
uFileListO
// 文件列表偏移
// 最大子文件个数
// 包文件的大小
// 包内文件信息结构
typedef struct SFilesMessage
// 本文件在包内的偏移
// 本文件的大小
szFileName[ 260 ]; // 本文件的路径
// 打开包文件
int OpenMasFile( const char* path, const byte onlyOpen )
// 写入文件信息次数;
bIsNew = 0;
// 是否新建的
// 文件头结构定义
g_pMasFile = fopen( path, "rb" );
// 用来判断是否存在
if ( g_pMasFile == NULL )
// 这里就没有用windows API了
if ( onlyOpen == 1 )
// 只打开不新建
return -1;
bIsNew = 1;
g_pMasFile = fopen( path, "wb" );
if ( g_pMasFile == NULL )
return -1;
// 先关闭,然后在用"rb+"方式打开
fclose( g_pMasFile );
g_pMasFile = fopen( path, "rb+" );
if ( g_pMasFile == NULL )
return -1;
if ( bIsNew == 1 ) // 新建的文件
header.uFileFlag
header.uFileCount
header.uFileListOfs
= sizeof( MaseHeader ); // 紧跟着就是文件列表
header.uMaxFileCount = MAX_FILE_COUNT;
header.uFileSize
= sizeof( MaseHeader )
+ ( MAX_FILE_COUNT * sizeof( FilesMsg ) );
// 写入头信息
fwrite( &header, sizeof( MaseHeader ), 1, g_pMasFile );
memset( &msg, 0, sizeof( FilesMsg ) );
uWriteCount = MAX_FILE_COUNT;
// 写入文件列表用0占位
while( uWriteCount-- )
fwrite( &msg, sizeof( FilesMsg ), 1, g_pMasFile );
// 文件存在
// 则读取头文件信息
fread( &header, sizeof( MaseHeader ), 1, g_pMasFile );
// 检查文件头标记
if ( header.uFileFlag != 'ESAM' )
fclose( g_pMasFile );
return -1;
// 检查数据是否完整
if ( header.uMaxFileCount != MAX_FILE_COUNT )
fclose( g_pMasFile );
return -1;
// 写文件到包里
int WriteFileToPak( const char* path )
// 此文件的文件信息结构
// 包文件头结构定义
uFileListEndO
pFile = NULL;
if ( g_pMasFile == NULL )
return -1;
memset( &fileMsg, 0, sizeof( FilesMsg ) );
fseek( g_pMasFile, 0, SEEK_SET );
// 则读取头文件信息
fread( &header, sizeof( MaseHeader ), 1, g_pMasFile );
uFileListEndOfs = header.uFileCount * sizeof( FilesMsg ) + header.uFileListO
pFile = fopen( path, "rb" );
if ( pFile == NULL )
return -1;
fseek( pFile, 0, SEEK_END );
uFileSize = ftell( pFile );
fseek( pFile, 0, SEEK_SET );
// 文件名长度不能超过260
strcpy( fileMsg.szFileName, path );
fileMsg.uFileOfs
= header.uFileS
fileMsg.uFileSize = uFileS
// 写入文件信息
// 将文件指针定位到uFileListEndOfs处,以便写入新的文件信息结构
fseek( g_pMasFile, uFileListEndOfs, SEEK_SET );
fwrite( &fileMsg, sizeof( FilesMsg ), 1, g_pMasFile );
// 申请空间
pBuff = ( byte* )malloc( uFileSize );
fread( pBuff, uFileSize, 1, pFile );
// 写数据到包文件里
fseek( g_pMasFile, header.uFileSize, SEEK_SET );
fwrite( pBuff, uFileSize, 1, g_pMasFile );
// 释放内存
free( pBuff );
// 重新填充header
header.uFileCount += 1;
header.uFileSize
fseek( g_pMasFile, 0, SEEK_SET );
// 重新写入包文件头
fwrite( &header, sizeof( MaseHeader ), 1, g_pMasFile );
// 从包文件里读数据
int ReadFileFromPak( const FilesMsg msg, byte* _dst )
if ( g_pMasFile == NULL )
return -1;
fseek( g_pMasFile, msg.uFileOfs, SEEK_SET );
fread( _dst, msg.uFileSize, 1, g_pMasFile );
// 获取包中某个文件的信息
int GetFileMessage( const char* path, FilesMsg* msg )
// 此文件的文件信息结构
// 包头结构
// 文件个数
if ( g_pMasFile == NULL || msg == NULL )
return -1;
// 则读取头文件信息
fseek( g_pMasFile, 0, SEEK_SET );
fread( &header, sizeof( MaseHeader ), 1, g_pMasFile );
uFileCount = header.uFileC
while ( uFileCount-- )
fread( &fileMsg, sizeof( FilesMsg ), 1, g_pMasFile );
// 判断是否是要获取的文件
if ( stricmp( fileMsg.szFileName, path ) == 0 )
*msg = fileM
return -1;
// 关闭包文件
int CloseMasFile( void )
if ( g_pMasFile == NULL )
return -1;
fclose( g_pMasFile );
g_pMasFile = NULL;
上面已经将整个打包解包接口给实现了,我自定义文件扩展名为.mase, 这个随意哈,文件头结构上面已经很清晰了。由于篇幅的原因,这里就不一一解说了,我贴了很多注释。应该能够看懂的。
有了上面的接口,我们就可以来操作这个包文件了,先是看怎么写入:
int main( void )
ret = OpenMasFile( "E://PhotoPak.mase", 0 );
if ( ret == -1 )
WriteFileToPak( "E://大山.jpg" );
WriteFileToPak( "E://海水.bmp" );
WriteFileToPak( "E://查看.exe" );
WriteFileToPak( "E://加载.dll" );
WriteFileToPak( "E://说明.txt" );
CloseMasFile();
在这段代码里,演示了怎么将文件给写进包文件,首先是创建了一个PhotoPak.mase包,然后是向里面写入了:大山.jpg、海水.bmp、查看.exe、加载.dll、说明.txt这么几个文件,注意我的接口里面都是用二进制打开的,因为如果是非二进制打开的话,写入的时候会插入一些物理字符(比如回车符(ASCII:0x0D( 1310&))等)。那样插入进去后,然后在解包时再采用非二进制方式写入文件就不是原来的文件了,这点大家要注意。
好了,写了这么几个文件后,再看看怎么把他们从包里面弄出来,然后能够正常的打开和查看:
int main( void )
ret = OpenMasFile( "E://PhotoPak.mase", 1 );
if ( ret == -1 )
ret = GetFileMessage( "E://查看.exe", &getFileMsg );
if ( ret == -1 )
pBuff = ( byte* )malloc( getFileMsg.uFileSize );
ret = ReadFileFromPak( getFileMsg, pBuff );
if ( ret == -1 )
goto __exit_
pOutFile = fopen( "E://查看_out.exe", "wb" );
// 注意使用的是二进制模式
if ( ret == -1 )
goto __exit_
fwrite( pBuff, getFileMsg.uFileSize, 1, pOutFile );
fclose( pOutFile );
__exit_free:
free( pBuff );
CloseMasFile();
很清楚了吧,直接先传入路径,然后获得文件的信息,方便我们分配空间。然后我是将从包里获取出来的文件又写到磁盘里,命名为查看_out.exe, 同样既然是获取了pBuff,你同样可以在内存中使用这个文件,一举两得。然后获取出来,运行这个获取的查看_out.exe看是不是能运行。我在WINDOWS XP SP3 下是能运行的,你可以用你自己的一个exe来测试,随便用什么文件。
这里还要说到几个注意事项:
1. 这里我只是测试了较小的文件解包和写包,如果文件比较大的话,可以用分块进行读写。
2. 我没有写任何的加密算法和压缩算法,这里只是展示了基本原理。也没有太多的效率和安全考虑。
3. 我这里使用的都是E盘根目录下的文件,你也完全可以不是跟目录,在包文件里面是没有文件夹的概念的,如果没有在根目录,你可以在解包的时候,根据路径先创建好文件夹在磁盘上,然后再将包里读出来的文件写到相应的路径下,这就实现了不同文件夹管理的功能。
阅读(...) 评论()

我要回帖

更多关于 linux编写定时任务 的文章

 

随机推荐