本文主要是介绍图书馆管理系统(4),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
之前说过写一个系统就等于写多个系统,
学生管理,成绩管理,餐饮管理,订票服务管理,银行存取钱财管理,通讯录号码管理
这些最基本的地方都是增删改查,再加一些别的函数功能等
在我们还没想出更多的模块功能时,还可以预留接口,后面想到了再加模块
加模块就是在主函数main这里后面再加case4,case5等
当代码出错调试时,哪个地方出错,就把断点下到哪个函数那儿
(下断点的前提是,先将下面的错误警告改完)
例如增加图书信息模块有错,断点就下到这
如果还有错,就将断点往前调
修改图书信息
来源,可右键点击转到该函数定义找到
来源
来源
修改图书编号时,有好几种修改情况
(1)当把1001再改成1001时,是可以的,也就相当于原来的编号没有动
(2)当想把1001改成1003时,就不行了,因为里面已经有一个存在的1003了;遵循编号唯一原则
(3)当想把1001改成1005时,是可以的,因为1005是新编号,里面没有;只是改完后里面就没有1001了
总结就是:
自己改自己,可以
改成重复的,不行
改成不存在的,可以
这3种情况都要写,前2种情况一致,都视为重复,然后不改编号,跳出循环,重新进
测试
(1)自己改成自己,编号不变,修改成功
1002改成1003
改完显示图书信息
修改成功,但返回后再重新显示的信息并未修改,这是因为——没有保存
保存函数
测试
修改函数
//保存修改后(iBookId)的数据
void SaveBookFile(int iBookId)
{FILE* pfBook;pfBook = fopen("book.txt","rb+");//只读,二进制,可追加if (pfBook != NULL)//正常打开,只把(iBookId)数据写进去{//要先将文件指针走到要修改(iBookId)的地方,再将该地方写进去(如1000条修改第555条)fseek(pfBook,LEN_BOOK*iBookId,SEEK_SET );//文件指针,偏移多少,初始位置if (fwrite(&astBook[iBookId],LEN_BOOK,1,pfBook)!=1)//写成功返回值1{printf("无法保存该信息!\n");}}fclose(pfBook);//忘关闭就会保存不成功
}//显示修改选项的菜单
void ShowModifyBookMenu()//显示修改的菜单
{printf("\n\n\n\n");printf("\t\t 1.编号 \n");printf("\t\t 2.书名 \n");printf("\t\t 3.作者 \n");printf("\t\t 4.出版社 \n");printf("\t\t 5.库存量 \n");printf("\t请输入所要修改的信息(输入相应的数字1-5):\n");}//修改图书信息
void ModifyBook()
{int iBookId;//要修改的图书编号int iBookRecord;//图书记录int iFlagExist;//该图书编号是否存在int iltem;//给选择(修改选项)int iNum;//图书编号int i;system("cls");//修改步骤//1.找到要修改的图书(看是否存在)iBookId = SearchBook();if (iBookId == -1)//没找到,但存在{printf("没找到该图书信息,但存在\n");return;}//如果没有图书信息(不存在)if (iBookId == -3){printf("没有该图书信息的记录,不存在\n");return;}//2.若存在,则修改图书信息iBookRecord =ReadBookFile("rb");//先打开文件,记录读了多少条信息//(一打开,内存数组就存了文件中书的信息),平时存放在外存,操作在内存//ReadBookFile函数将文件中的读到内存数组里(读:后读到前里面;写:前写到后里面)//给提示,要修改的内容是什么(所有的图书信息就是结构体里面封装的那些,里面的都可修改)//给一个选择内容界面ShowModifyBookMenu();//显示修改的菜单scanf("%d" ,& iltem);getchar();//修改哪一个switch (iltem){case 1://修改图书编号,在输入新的图书编号之前要判断你输入的编号是否已经存在printf("请输入新改的图书编号:");do{iFlagExist = 0;scanf("%d",&iNum);//输入新改编号getchar();for (i = 0; i < iBookRecord;i++)//将内存中数组的所有图书编号遍历一遍,看是否与新改的重复{if (iNum == astBook[i]. iNum&& i!=iBookId)//相同,重复且不存在{iFlagExist = 1;//重新进doprintf("错误,或该图书编号已经存在,请重新输入:");break;}}} while (iFlagExist == 1);//先一次do,符号while进doastBook[iBookId].iNum = iNum;//赋值新改编号break;case 2://修改图书名称printf("请输入新改的图书名称:");scanf("%s", &astBook[iBookId].acName);getchar();//这里不用判断直接赋值,因为图书的名称相同或重复也可以,不像编号要唯一确定一本书,//当然细致判断一下也可以,就是自己改自己,判断不做,不判断做,的结果都一样break;case 3://修改图书作者printf("请输入新改的图书作者:");scanf("%s", &astBook[iBookId].acAuthor);getchar();break;case 4://修改图书出版社printf("请输入新改的图书出版社:");scanf("%s", &astBook[iBookId].acPress);getchar();break;case 5://修改图书库存量printf("请输入新改的图书库存量:");scanf("%d", &astBook[iBookId].iAmout);getchar();break;}//在数组中改完保存到文件,保存函数//todo//12345修改完统一保存SaveBookFile(iBookId);//找到(Search)的几号id的书,修改完,再将它保存printf("图书信息修改成功!\n");
}
测试
现在图书管理模块就写完了
#define _CRT_SECURE_NO_WARNINGS //这一句必须放在第一行
#include<stdio.h>
#include<Windows.h>
#include <conio.h>#define BOOK_DATA astBook[i].iNum,astBook[i].acName,astBook[i].acAuthor,astBook[i].acPress,astBook[i].iAmout
//struct Book astBook[BOOK_NUM];图书信息用每一个结构体数组的下标来表示
#define BOOK_NUM 200
#define READER_NUM 100
#define LEN_BOOK sizeof(struct Book)//一次写一本书的全部(整个结构体)信息,用结构体封装//用结构体封装图书信息
struct Book
{int iNum;//图书编号char acName[15];//图书名称char acAuthor[15];//图书作者char acPress[15];//图书出版社int iAmout;//(某本)库存量,为借书,还书,增减做准备
};//用结构体封装读者信息,信息种类由自己决定,可增删
struct Reader
{int iNum;//读者编号char acName[15];//读者姓名char acSex[4];//读者性别int iMax;//读者最大可借读数量int iAmout;//读者当前可借阅的数量int Bookld[10];//读者已经借阅的读书列表,例如写10即一个人最多可已经借了10本书
};//定义2个全局结构体数组,用来记录所有的图书信息和读者信息
struct Book astBook[BOOK_NUM];
struct Reader astReader[READER_NUM];void ShowMainMeun()//图书馆管理系统的主界面
{system("cls");//清屏函数printf("\n\n\n\n\n\n");//将字体放在页面居中位置,\n别太靠上printf("\t\t**********欢迎进入***********\n");//\t别太靠左printf("\t\t**********图书馆管理系统*****\n");printf("\t\t**********主菜单**************\n");printf("\t\t**********1.图书管理**********\n");printf("\t\t**********2.读者管理**********\n");printf("\t\t*********3.借书还书登记*******\n");printf("\t\t**********0.退出系统**********\n");printf("\t\t请选择:0-3");printf("\n");
}void ShowBookMunu()//显示书籍管理的子菜单
{//显示菜单界面的函数一进来都先清屏system("cls");//清屏函数printf("\n\n\n\n\n\n");printf("\t\t**************欢迎进入**************\n");printf("\t\t**********图书馆管理系统***********\n");printf("\t\t***************子菜单***************\n");printf("\t\t**********1.显示图书信息***********\n");printf("\t\t**********2.增加图书信息***********\n");printf("\t\t**********3.查询图书信息***********\n");printf("\t\t**********4.删除图书信息***********\n");printf("\t\t**********5.修改图书信息***********\n");printf("\t\t**********0.返回主菜单*************\n");printf("\t\t请选择:0-5");printf("\n");
}//从文件中读取数据
int ReadBookFile(const char* pcMode)//以什么方式(pcMode)去读,
//显示信息中是以只读的方式打开文件,这里相当于函数的复用,pcMode==rb
{int iBookRecord = 0;//文件中的记录条数FILE* pfBook;//文件指针pfBook = fopen("book.txt", pcMode);//打开文件,路径为当前文件的路径,即相对路径if (pfBook == NULL){printf("文件打开失败\n");return -1;}//把数据读到上面定义的全局结构体数组里面去while (!feof(pfBook))//要是文件末尾,则feof(pfBook)为非0,文件里为0{if (fread(&astBook[iBookRecord], LEN_BOOK, 1, pfBook))//从pfBook中不断读取的数据,存放到astBook结构体里面去,每次读的大小为LEN_BOOK,数量为1{iBookRecord++;}}fclose(pfBook);return iBookRecord;//返回图书条数}void ShowBook()//显示图书信息
{int i, iBookRecord;//信息条数,有多少已经记录了的图书,//也就是现在的图书信息里面有多少条,比如一个book文件里面就存了1000条图书信息system("cls");//显示函数均清屏iBookRecord = ReadBookFile("rb");//rb以只读的方式打开文件, ReadBookFile是一个从文件中读的函数if (iBookRecord == -1)//ReadBookFile函数失败返回-1{printf("该文件打开失败,请先新增图书信息!\n");}if (iBookRecord == 0)//刚开始book是一个空文件{printf("文件中没有图书信息!\n");}else//依次打印输出所有的图书信息,对照封装图书信息的结构体来打印{printf("\t**********************************\n");printf("\n\n");printf("\t\t%-6s%-16s%-10s%-20s%-4s\n","编号","书名","作者","出版社","库存量");//数字前面加负号为左对齐,都是%s是因为刚开始打印的是抬头汉字//现在依次打印图书信息for (i = 0; i < iBookRecord; i++){printf("\t\t%-6d%-16s%-10s%-20s%-4d\n",BOOK_DATA);//太长的内容为了防止反复写就用宏代替}printf("\t**********************************\n");}
}void AddBook()//新增图书信息
{//图书信息是往文件里面增加的,每次打开文件要往已有记录的后面新加,//所以要以追加的方式打开,不能以w打开,w打开会删除原有数据FILE* pfBook;//文件指针int iBookRecord;//图书记录条数int iFlagExist;//该图书是否存在,录入时要用int i;//结构体数组下标char cFlag;//是否继续输入的标志system("cls");//清屏iBookRecord = ReadBookFile("ab+");//ab+是以追加方式打开或新建二进制文件。//这里函数复用,pcMode==ab+if (iBookRecord == -1){printf("文件打开失败\n");return ;}else if (iBookRecord == 0){printf("文件为空,没有图书记录!\n");}else{ShowBook();//记录不为0则显示现有图书记录信息}//循环录入图书信息,一次录完所有图书信息//不想循环就要多次进入,录一本书进入一次printf("请选择是否输入信息(y/n)或(Y/N)\n");cFlag = getchar();//getchar将刚才从键盘读取的字符(y/n)丢给cFlaggetchar();//去掉\nif (cFlag == 'n'|| cFlag == 'N')//选择否,一般没有提示,直接退出(到上一层){return;}//打开pfBook文件信息,方便后面保存信息pfBook = fopen("Book.txt", "ab +");if (pfBook == NULL){printf("文件打开失败\n");return;}while(cFlag == 'y'|| cFlag == 'Y'){printf("录入数据:\n");//录入数据,往内存中的结构体数组录入数据//定义录入的数组的大小,先判断录入的数据不能超过数组大小,这里宏定义是200,//即判断记录是否已满if (iBookRecord >= BOOK_NUM){printf("记录已满!\n");fclose(pfBook);return;}// 要考虑图书编号是唯一的,不能重复;printf("请输入图书编号:\n");//不能输一个编号就退出,要循环录do whiledo{iFlagExist = 0;scanf("%d", &astBook[iBookRecord].iNum);getchar();//去掉编号的\n//循环挨个对比,编号是否重复for (int i = 0;i < iBookRecord; i++){//iFlagExist= 0;//这里置0会出错if (astBook[i].iNum == astBook[iBookRecord].iNum)//重复,数组已存在==新输入{iFlagExist = 1;printf("该图书编号已经存在,请重新输入:");break;}}} while(iFlagExist == 1);//重复了重新进入do循环,输入新编号;//要是不为1为0,就是正常的,就不重复退出当前do循环//输入图书信息printf("请输入图书名称:");scanf("%s", astBook[iBookRecord].acName);getchar();printf("请输入图书作者:");scanf("%s", astBook[iBookRecord].acAuthor);getchar();printf("请输入图书出版社:");scanf("%s", astBook[iBookRecord].acPress);getchar();printf("请输入图书库存量:");scanf("%d", &astBook[iBookRecord].iAmout);getchar();//录完数据后保存信息,将信息保存在外存的文件中//从数组存到文件//这里不太好的地方是,它是一条一条保存的,即录完一条当即保存一条//这个保存也可以封装成一个函数,后面增删改查等所有操作执行完都需要保存,直接调用函数会更方便,不然会发现保存的代码会一直重复的写if (fwrite(&astBook[iBookRecord], LEN_BOOK, 1, pfBook) != 1)//把buffer里面的数据给他写,写到FILE*里面去{printf("无法保存该信息!\n");return;}else//fwrite返回1,即当前一个信息读完了{printf("%d号图书信息已经保存!\n", astBook[iBookRecord].iNum);iBookRecord++;}//给提示是否还录printf("还要继续输入信息吗?(y/n)或(Y/N)\n");cFlag = getchar();getchar();}fclose(pfBook);printf("添加图书信息执行完毕\n");
}//查找查询图书信息
int SearchBook()
{//将文件以只读的方式打开,在里面一个个找就是查询了//按编号找,因为编号是唯一不重复的int iBookNum;//图书编号int iBookReacord;//图书记录条数int iBookId;//返回值,查到该书了就返回其返回值int i;system("cls");iBookReacord = ReadBookFile("rb");//ReadBookFile读文件函数,每回将以什么方式读的参数传进去就可以了if (iBookReacord == -1){printf("文件打开失败!\n");printf("按任意键返回到子菜单");getchar();return -2;//文件打开失败,返回-2,//这里有很多种返回情况,每种情况的返回值不一样好区分,都写return难排错//方便后面排错,类似于一张表上-1对应的错,-2的,-3的错误分类}else if (iBookReacord == 0){printf("没有图书记录!\n");printf("按任意键返回到子菜单");getchar();return -3;//没有图书记录,返回-3}//进入查找程序printf("请输入想要查找的图书编号:");scanf("%d", &iBookNum);getchar();//在文件里循环,挨个查找有没有该图书编号for (i = 0; i < iBookReacord; i++)//一共三种编号种类的书,循环到2下标{if (iBookNum == astBook[i].iNum)//编号相等{iBookId = i;//图书返回值为该书在数组中的下标位置printf("\t\t%d号图书信息如下:", iBookNum);printf("\t\t------------------------------\n");printf("\t\t%-6s%-16s%-10s%-20s%-4s\n", "编号", "书名", "作者", "出版社", "库存量");//数字前面加负号为左对齐,都是%s是因为刚开始打印的是抬头汉字printf("\t\t%-6d%-16s%-10s%-20s%-4d\n", BOOK_DATA);printf("\t\t------------------------------\n");break;}}if (i == iBookReacord)//出for循环未进入if{printf("找不到%d号图书信息!\n",iBookNum);iBookId = -1;//找不到记录,返回-1}return iBookId;
}//删除图书信息
void DeleteBook()
{FILE* pfBook;int iBookId;//要删除的图书编号int iBookRecord;//图书记录int i;char cFlag;//是否删除标志system("cls");iBookId = SearchBook();//找到该图书idif (iBookId == -1)//没找到信息{return;}iBookRecord = ReadBookFile("rb");//只读打开看文件里的图书记录条数printf("已经找到该图书,是否删除?[y(Y)/n(N)]");cFlag = getchar();//读y/ngetchar();//丢\n//删除之前先保存删除之前的图书编号,因为删除之后图书信息会被覆盖int index = astBook[iBookId].iNum;//绿色位置id的号码numif (cFlag == 'n' || cFlag == 'N')//选择否,一般没有提示,直接退出(到上一层){return;}else if (cFlag == 'y' || cFlag == 'Y'){//删除找到的书(将后面的书依次往前移动,穿透删除)for (i = iBookId; i < iBookRecord-1; i++)//i=iBookId找到的绿色位置,i < iBookRecord,i<红色末尾{astBook[i] = astBook[i + 1];//,数组依次前移,后面值覆盖前面值//i+1可能越界,所以 iBookRecord-1,或者i+1< iBookRecord}//删除完记录条数(红色)减1iBookRecord--;}//删除完成保存现有修改后的信息//现在删掉的是数组里面的信息(在数组里操作),文件里面的还未删除//数组里删完之后整体复制到文件中,保存到文件pfBook = fopen("book.txt", "wb");//wb删除原有打开新空白if (pfBook != NULL)//指针不空复制写进文件去,写完关文件{for (i = 0; i < iBookRecord; i++){if (fwrite(&astBook[i], LEN_BOOK, 1, pfBook) != 1)//写成功完返回写的条数1{printf("无法保存该信息!\n");return;}}fclose(pfBook);printf("%d号图书已经删除!\n",index);}}//保存修改后(iBookId)的数据
void SaveBookFile(int iBookId)
{FILE* pfBook;pfBook = fopen("book.txt","rb+");//只读,二进制,可追加if (pfBook != NULL)//正常打开,只把(iBookId)数据写进去{//要先将文件指针走到要修改(iBookId)的地方,再将该地方写进去(如1000条修改第555条)fseek(pfBook,LEN_BOOK*iBookId,SEEK_SET );//文件指针,偏移多少,初始位置if (fwrite(&astBook[iBookId],LEN_BOOK,1,pfBook)!=1)//写成功返回值1{printf("无法保存该信息!\n");}}fclose(pfBook);//忘关闭就会保存不成功
}//显示修改选项的菜单
void ShowModifyBookMenu()//显示修改的菜单
{printf("\n\n\n\n");printf("\t\t 1.编号 \n");printf("\t\t 2.书名 \n");printf("\t\t 3.作者 \n");printf("\t\t 4.出版社 \n");printf("\t\t 5.库存量 \n");printf("\t请输入所要修改的信息(输入相应的数字1-5):\n");}//修改图书信息
void ModifyBook()
{int iBookId;//要修改的图书编号int iBookRecord;//图书记录int iFlagExist;//该图书编号是否存在int iltem;//给选择(修改选项)int iNum;//图书编号int i;system("cls");//修改步骤//1.找到要修改的图书(看是否存在)iBookId = SearchBook();if (iBookId == -1)//没找到,但存在{printf("没找到该图书信息,但存在\n");return;}//如果没有图书信息(不存在)if (iBookId == -3){printf("没有该图书信息的记录,不存在\n");return;}//2.若存在,则修改图书信息iBookRecord =ReadBookFile("rb");//先打开文件,记录读了多少条信息//(一打开,内存数组就存了文件中书的信息),平时存放在外存,操作在内存//ReadBookFile函数将文件中的读到内存数组里(读:后读到前里面;写:前写到后里面)//给提示,要修改的内容是什么(所有的图书信息就是结构体里面封装的那些,里面的都可修改)//给一个选择内容界面ShowModifyBookMenu();//显示修改的菜单scanf("%d" ,& iltem);getchar();//修改哪一个switch (iltem){case 1://修改图书编号,在输入新的图书编号之前要判断你输入的编号是否已经存在printf("请输入新改的图书编号:");do{iFlagExist = 0;scanf("%d",&iNum);//输入新改编号getchar();for (i = 0; i < iBookRecord;i++)//将内存中数组的所有图书编号遍历一遍,看是否与新改的重复{if (iNum == astBook[i]. iNum&& i!=iBookId)//相同,重复且不存在{iFlagExist = 1;//重新进doprintf("错误,或该图书编号已经存在,请重新输入:");break;}}} while (iFlagExist == 1);//先一次do,符号while进doastBook[iBookId].iNum = iNum;//赋值新改编号break;case 2://修改图书名称printf("请输入新改的图书名称:");scanf("%s", &astBook[iBookId].acName);getchar();//这里不用判断直接赋值,因为图书的名称相同或重复也可以,不像编号要唯一确定一本书,//当然细致判断一下也可以,就是自己改自己,判断不做,不判断做,的结果都一样break;case 3://修改图书作者printf("请输入新改的图书作者:");scanf("%s", &astBook[iBookId].acAuthor);getchar();break;case 4://修改图书出版社printf("请输入新改的图书出版社:");scanf("%s", &astBook[iBookId].acPress);getchar();break;case 5://修改图书库存量printf("请输入新改的图书库存量:");scanf("%d", &astBook[iBookId].iAmout);getchar();break;}//在数组中改完保存到文件,保存函数//todo//12345修改完统一保存SaveBookFile(iBookId);//找到(Search)的几号id的书,修改完,再将它保存printf("图书信息修改成功!\n");
}void MangerBook()//图书管理模块,实现增删改查和显示
//跟主菜单类似要给一个界面显示,里面给用户选择
{int iltem;//给用户的选择ShowBookMunu();//显示书籍管理的子菜单scanf("%d", &iltem);getchar();//去掉scanf里面iltem的\n,如果没有while循环读它就可以不写while (iltem)//进到子菜单里面{switch (iltem){case 1:ShowBook();//显示图书信息break;case 2:AddBook();//新增图书信息break;case 3:SearchBook();//查找图书信息break;case 4:DeleteBook();//删除图书信息break;case 5:ModifyBook();//修改图书信息break;default:printf("\t\t请输入正确的数字!\n\t\t");}//返回主菜单按0退到上一层就好//返回当前子菜单printf("按任意键返回子菜单");_getch();//从控件中获取字符而无需回显字符。
//#include <conio.h>控制台的输入输出流,getch加_是因为之前的不安全ShowBookMunu();scanf("%d", &iltem);//进while循环getchar();}
}int main(){ShowMainMeun();//显示主函数的界面,图书馆管理系统的主界面int iltem;//用户输入的选择数字,用scanf读进去scanf("%d", &iltem);//%d后面不能有空格while (iltem)//0退出,非0为真,进入选择模块{switch (iltem){case 1://todo是为了提醒这个模块函数需要写但还没写,后面找todo来写MangerBook();//图书管理模块,没实现的模块先屏蔽起来break;case 2://todo//MangerReader();//读者管理模块break;case 3://todo//BorrowReturnManger();//借书还书模块break;default://除了0123之外的数字的结果printf("\t\t请输入正确的数字!\n\t程序将在3秒后跳转到主菜单");Sleep(3000);//停留休眠3秒,在Windows里面它的单位是毫秒,1秒=10^3毫秒}//出循环回到(显示)主菜单,即ShowShowMainMeun();scanf("%d", &iltem);getchar();//读scanf的\n,再丢掉,避免while (iltem)读进去}return 0;}
一般进到下一层模块,就一个Switch,然后case不同情况
然后就是文件,和内存的意识
例如:显示界面——就是文件读到内存
而数据的所有操作函数等,都是在内存中的数组(astBook)里进行操作,因为文件IO里面的操作特别慢
所以每一个函数里面,都有打开文件,读到内存的操作
然后操作完就需要保存函数,存回文件
如果不单独写保存操作,直接调用保存函数的话
就要写两个版本的保存函数,一个是动文件指针位置(光标)的fseek
一个是不动文件指针位置的
那么
这篇关于图书馆管理系统(4)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!