c语言课程设计:图书管理系统设计的基本思路是什么?

Python019

c语言课程设计:图书管理系统设计的基本思路是什么?,第1张

图书管理系统主要要求可以录入书籍,添加书目,查找书本信息,删除或修改信息,有的还要求显示是否被借阅等。

一般采用结构体数组,链表,文件操作和自定义函数。主要是需要对基础知识掌握牢固。

先定义结构体,然后对结构体的成员进行定义,选择数组存储书本各种信息。录入信息可以用for和do while循环等来做。

存放信息需要文件操作函数,比如fopen,fwrite等。

删除和添加可以删除节点或者增加节点。

查找之类的可以用字符串操作的各种函数实现。

附上参考源代码

#include <stdio.h>

#include <string.h>

#include <stdlib.h>

#include <conio.h>

#define  books "f:\\books.txt"

#define booksbak  "f:\\booksbak.txt"

struct bookinfo

{

char isbn[20]

char title[30]

char author[20]

int count

}

struct book

{

struct bookinfo onebook

struct book *next

}

struct book *searchBook ( struct book *listptr,char isbn[])

{

while(listptr!=(struct book *)0)

if (strcmp(listptr->onebook.isbn,isbn)==0)

return listptr

else

listptr=listptr->next

return (struct book *)0

}

void MainSearchbook(struct book *firstptr)

{

struct book *ptr

char isbnno[20]

printf("请输入ISBN:")

scanf("%s",&isbnno)

ptr=searchBook(firstptr,isbnno)

if (ptr!=(struct book *)0)

{

printf("找到了!!!\n")

printf("ISBN:%s\n",ptr->onebook.isbn)

printf("Title:%s\n",ptr->onebook.title)

printf("Author:%s\n",ptr->onebook.author)

}

else

printf("sorry,not found!!!\n")

}

int addBook(struct book *listptr,struct bookinfo note)

{

while(listptr->next!=0)

listptr=listptr->next

listptr->next=(struct book *)malloc(sizeof(struct book))

listptr->next->onebook=note

listptr->next->next=0

return 0

}

void MainAdd(struct book *listptr,FILE *fp)

{

int ok

struct bookinfo note

printf("请输入ISBN:")

scanf("%s",¬e.isbn)

printf("请输入Title:")

scanf("%s",¬e.title)

printf("请输入Author:")

scanf("%s",¬e.author)

ok=addBook(listptr,note)

if (ok==0)

{

//将加入的图书写到文件中保存

fprintf(fp,"\n%s %s %s %d",note.isbn,note.title,note.author,0)

printf("添加图书成功!!!\n")

}

else

printf("添加图书失败!!!\n")

}

int removeBook(struct book *listptr,char isbn[])

{

while(listptr->next!=(struct book *)0)

{

if (strcmp(listptr->next->onebook.isbn,isbn)==0)

{

listptr->next=listptr->next->next

return 0

}

else

listptr=listptr->next

}

return -1

}

void MainRemove(struct book *listptr,FILE *fp)

/************************************************************************/

/* 删除书籍函数,通过ISBN删除链表节点,同时删除文件中对应信息              */

/* 删除文件中一行,用的是笨方法,把需要的信息写到新文件,删除旧文件,重命名..*/

/************************************************************************/

{

char isbnno[20]

int ok

struct bookinfo onebook

printf("请输入ISBN:")

scanf("%s",&isbnno)

ok=removeBook(listptr,isbnno)

if (!ok)

{

FILE *fpbak

if ((fpbak=fopen(booksbak,"a+"))==NULL)

printf("文件打开失败!!!\n")

fseek(fp,0,SEEK_SET)   //移到文件开始

while((fscanf(fp,"%s %s %s %d\n",&onebook.isbn,&onebook.title,&onebook.author,&onebook.count))!=EOF)

{

if (strcmp(onebook.isbn,isbnno)!=0)

{

fprintf(fpbak,"%s %s %s %d\n",onebook.isbn,onebook.title,onebook.author,onebook.count)

}

}

fclose(fp)

fclose(fpbak)

if (remove(books))   //删除失败返回非0

{

printf("删除文件失败!!!\n")

return 

}

else

if (rename(booksbak,books))  //重命名失败返回非0值

{

printf("重命名失败!!!\n")

return 

}

printf("删除成功!!!\n")

}

else

printf("查无此书!!!")

}

int  choice(void)

{

int c

printf("1.查看图书\n")

printf("2.添加图书\n")

printf("3.删除图书\n")

printf("4.退出程序\n")

printf("请选择序号:")

return c=getchar()

//return c=getche()

printf("\n\n")

}

int addEntry(FILE *fp,struct book *firstptr)

/************************************************************************/

/*        主要用来加载文件中存放的图书信息                              */

/************************************************************************/

{

struct bookinfo onebook

while((fscanf(fp,"%s %s %s %d\n",&onebook.isbn,&onebook.title,&onebook.author,&onebook.count))!=EOF)

{

while(firstptr->next!=0)

firstptr=firstptr->next

firstptr->next=(struct book *)malloc(sizeof(struct book))

firstptr->next->onebook=onebook

firstptr->next->next=0

}

return 0

}

int main(int argc,char *argv[])

{

int ch

struct book first

strcpy(first.onebook.isbn,"123456")

strcpy(first.onebook.title,"Programming C")

strcpy(first.onebook.author,"yhb")

first.next=0

struct book *firstptr=&first   //链表头指针

FILE *fp

if ((fp=fopen(books,"a+"))==NULL)

printf("文件打开失败!!!")

addEntry(fp,firstptr)

while(1)

{

    system("CLS")   //清屏

/************************************************************************/

/*          想想这里为什么要清空缓冲区?                                 */

/*由于上一次(choice函数)的getchar(),还有一个'\n'留在缓冲区....          */

/*可以把这句话注释掉看看,没有这句话会遇到麻烦                           */

/*如果不用fflush,可以将上面的getchar()换成getche()                      */

/*比较getchar(),getch(),getche()......                                  */

/************************************************************************/

fflush(stdin) 

ch=choice()-48

switch (ch)

{

case 1:

MainSearchbook(firstptr)

break

case 2:

MainAdd(firstptr,fp)

break

case 3:

MainRemove(firstptr,fp)

break

case 4:

printf("谢谢使用...\n")

exit(0)

default:

printf("请输入正确序号!")

}

system("PAUSE")

}

return 0

}

一 程序设计说明书

【设计题目】 图书馆借阅管理

【问题描述】图书馆,适合用C++面向对象的功能来描述。图书馆管理系统分为借书、还书、图书管理和读者服务等四个部分。设计一个读者类Reader,记录每个读者基本信息;读者库类Rdatabase,记录所有读者信息;图书类Book, 记录每本书的基本信息;图书库类Bdatabase, 记录所有图书信息。

【基本要求】

1读者库类RDatabase中,其构造函数中,将read.txt文件中所有读入读者记录rede[]中。处理完毕,在析构函数中将read[]中的所有未删记录写入到read.txt中。

2图书库类BDatabase中,其构造函数中,将book.txt文件中所有读入图书记录book[]中。处理完毕,在析构函数中将book[]中的所有未删记录写入到book.txt中。

3 利用构造函数完成读者和图书数据初始化,完成内存分配。程序结束前,析构函数完成所申请的堆内存空间。

4 编写主函数,对所编写的矩阵类进行全面测试。要求用户界面采用菜单方式。测试中需要读者和图书数据通过I/O流从磁盘文件读入,同时显示在屏幕上。得到的数据通过I/O流写入磁盘文件保存,同时显示在屏幕上。

5 源程序中要有充分的注释,报告中要有详细的流程图和文字材料。

【类的设计】

该程序包含了四个类,如下:

1.Reader类,有读者的基本管理功能,具有以下私有数据:

int tag//删除标记 1:已删;0:未删

int no//读者编号

char name[10]//读者姓名

int borbook[Maxbor]//所借图书

2.读者库类Rdatabase, 具有以下私有数据:

int top//读者记录指针

Reader read[Maxr]//读者记录

3.图书库类Book,有一本图书的基本功能,有以下私有数据:

int tag//删除标记 1:已删;0:未删

int no//图书编号

char name[20]//书名

int onshelf//是否在架 1在架 0已借

4.图书库类BDatabase,有以下私有数据:

int top//图书记录指针

Book book[Maxb]//图书记录

【特殊函数的设计说明】

构造函数

1.Reader类中构造函数Reader(),初始化函数;

2.读者库类RDatabase中,其构造函数Rdatabase(),将read.txt文件中所有读入读者记录rede[]中。

3.Book类中构造函数Book(),初始化函数;

4.图书库类BDatabase中,其构造函数中,将book.txt文件中所有读入图书记录book[]中。

拷贝构造函数

Reader类中的拷贝构造函数将getname()的返回值读者姓名拷贝到setname()中,Book类中的拷贝构造函数将getname()函数的返回值图书名拷贝到图书名设置函数setname()中。

析构函数

1.读者库类RDatabase中,其析构函数~Rdatabase(),将read[]中的所有未删记录写入到read.txt中;

2.图书库类BDatabase中,其析构函数~Bdatabase(),将book[]中的所有未删记录写入到book.txt中。

运算符重载

重载了“=”,当比较读者编号和图书编号时,重载;重载位运算符“〈〈”和“〉〉”等。

【主要函数算法流程图】

【程序的使用方法】

1.进入操作页面,按提示操作;

2.首先,新增图书和读者信息,之后就可以对以存在的信息进行操作;

3.操作当中,可以随时增加,更改和删除图书或读者信息;

4.当选择退出时,进行清屏。

二 程序上机调试报告

【语法错误及其排除】

1.在敲程序时,有很多拼写错误,例好多处把Readdata()误打Readdate();结束的分号,在不同的输入法状态下输入,这些小错误刚开始很难发现,不过有了经验,就很容易了。

2.创建新的构造函数时,使用出现了错误。重载构造函数要注意函数的访问权限,结果就不会出现错误。

【算法错误及其排除】

1.读者类中借书操作函数中,采用循环语句时判断读者已借图书量时for(int i=0i<Maxbori++)误写为for(int i=1i<Maxbori++),使循环发生错误。

2.指针使用错误,指针b和r混淆,导致编译错误得到“error C2660: 'retbook' : function does not take 1 parameters”错误报告。

#include<stdio.h>

#include<string.h>

#include<stdlib.h>

struct book{

char ISBN[10]

char book[30]

char author[20]

char edition[10]

char press[50]

char year[10]

}

void add_book()

void delete_book()

void modify_book()

void scan_book()

void Me()

int main()

{

int c3

while(1)

{

system("color 2C")

printf("\t\t\t=========图书管理系统======\n")

printf("\t\t\t [1]添加图书 \n")

printf("\t\t\t [2]删减图书 \n")

printf("\t\t\t [3]修改图书信息\n")

printf("\t\t\t [4]浏览图书信息\n")

printf("\t\t\t [5]作者信息 \n")

printf("\t\t\t [6]退出图书管理系统 \n")

printf("\t\t\t=========================\n")

printf("\n")

printf("请选择你的操作 :")

scanf("%d",&c3)

getchar()

switch(c3)

{

case 1:add_book()break

case 2:delete_book()break

case 3:modify_book()break

case 4:scan_book()break

case 5:Me()break

case 6:system("cls")

//return 0这是返回就跳出程序了,会终止程序运行

}

}

return 0//其实这句被执行,这个程序就终结了,释放所有被使用的资源。

}

//添加

void add_book()

{

FILE *fp

struct book n

struct book nn

char x

int l,r

fp=fopen("file:///Users/liangyiwen/Desktop/Book/图书管理系统.txt","ab+")

do

{

system("cls")

do//I标志用得不好

{

// l=0此处的移进下一循环内

printf("\n")

printf("编号 书名 作者 版本号 出版社 出版时间\n")

fflush(stdin)

scanf("%s %s %s %s %s %s",n.ISBN,n.book,n.author,n.edition,n.press,n.year)

system("cls")

rewind(fp)

while(!feof(fp))

{

l=0// 移进此处,每次判断都是设定默认值,否则被修改后不是默认值

fread(&nn,sizeof(nn),1,fp)

r=strcmp(n.book,nn.book)

if(r==0)

{

l=1

printf(" 该图书已存在,请输入新的图书信息: \n\n")

break

}

}

}while(l)

fwrite(&n,sizeof(n),1,fp)

printf("\n")

printf("是否继续输入新的图书信息[y/n] \n\n")

do

{

x=getchar()

}while(x!='n'&&x!='y')

}while(x=='y')

fclose(fp)

system("cls")

}

//删除图书

void delete_book()

{

FILE *fp,*fp1

char x,z

struct book n

struct book nn

int l,r

fp=fopen("file:///Users/liangyiwen/Desktop/Book/图书管理系统.txt","ab+")

while(1)

{

l=0

system("cls")

printf("\n")

printf("请输入需要删除的图书名: \n\n")

fflush(stdin)

scanf("%s",&n.book)

rewind(fp)

while(1)

{

fread(&nn,sizeof(nn),1,fp)

if(feof(fp))

break

r=strcmp(n.book,nn.book)

if(r==0)//本来可以 得到结果后可以直接进行删除动作,没有则检查下一个,你现在

//不用这种结构,写得太多了,不跟你修改了

{

l=1

break

}

if(l==0)

{

printf("\n")

printf("对不起,未找到该图书 \n")

}

else

{

printf("\n")

printf("编号 书名 作者 版本号 出版社 出版时间\n")

printf("\n")

//下面应该是打印,n的值应该从搜索到值读出来打印,怎么还手工输入图书信息,手

//工输入能保证每一次输入都与原信息一样吗?

scanf("%s %s %s %s %s %s",n.ISBN,n.book,n.author,n.edition,n.press,n.year)

printf("\n")

printf("确认删除图书信息? [是(y)/否(n)]\n\n")

do

{

z=getchar()

}while(z!='n'&&z!='y')

if(z=='n')

break

else

{

fp1=fopen("file:///Users/liangyiwen/Desktop/Book/图书管理系统.txt","wb")

rewind(fp)

while(1)

{

fread(&nn,sizeof(nn),1,fp)

if(feof(fp))

break

r=strcmp(n.book,nn.book)

if(r!=0)

fwrite(&nn,sizeof(nn),1,fp1)

}

fclose(fp)

fclose(fp1)

fp=fopen("file:///Users/liangyiwen/Desktop/Book/图书管理系统.txt","wb")

fp1=fopen("file:///Users/liangyiwen/Desktop/Book/图书管理系统new.txt","rb")

while(1)

{

fread(&nn,sizeof(nn),1,fp1)

if(feof(fp1))

break

fwrite(&nn,sizeof(nn),1,fp)

}

fclose(fp)

fclose(fp1)

}

}

printf("\n")

printf("是否继续删除图书信息? [是(y)/否(n)] \n")

do

{

x=getchar()

}while(x!='n'&&x!='y')

if(x=='n')

break

}

fclose(fp)

system("cls")

}

//修改图书信息

void modify_book()

{

FILE *fp

struct book n

struct book nn

int l,r

char x

fp=fopen("file:///Users/liangyiwen/Desktop/Book/图书管理系统.txt","rb+")

while(1)

{

l=0

printf("\n")

system("cls")

printf("请输入需要修改的图书名: \n\n")

fflush(stdin)

scanf("%s",&n.book)

system("cls")

rewind(fp)

while(1)

{

fread(&nn,sizeof(nn),1,fp)

if(feof(fp))

break

r=strcmp(n.book,nn.book)

if(r==0)

{

l=1

break

}

}

if(l==0)

{

printf("\n")

printf("对不起,未找到该图书信息 \n\n")

}

else

{

printf("\n")

printf("编号 书名 作者 版本号 出版社 出版时间\n")

printf("\n")

//下面应该是打印,n的值应该从搜索到值读出来打印,怎么还手工输入图书信息,

//手工输入能保证每一次输入都与原信息一样吗?

scanf("%s %s %s %s %s %s",n.ISBN,n.book,n.author,n.edition,n.press,n.year)

printf("请依次修改图书信息\n\n\n")

fflush(stdin)

scanf("%s %s %s %s %s %s",n.ISBN,n.book,n.author,n.edition,n.press,n.year)

fseek(fp,sizeof(nn),1)

fwrite(&n,sizeof(nn),1,fp)

}

printf("\n")

printf(" 是否继续修改用户信息[y/n]? \n\n")

do

{

x=getchar()

}while(x!='n'&&x!='y')

if(x=='n')

break

}

fclose(fp)

system("cls")

}

//浏览

void scan_book()

{

FILE*fp

char x

struct book n

fp=fopen("file:///Users/liangyiwen/Desktop/Book/图书管理系统.txt","rb")

rewind(fp)

system("cls")

while(1)

{

fread(&n,sizeof(n),1,fp)

if(feof(fp))

break

else

{

printf("编号 书名 作者 版本号 出版社 出版时间\n")

printf("\n")

//下面应该是打印,n的值应该从搜索到值读出来打印,阅览信息怎会输入信息

scanf("%s %s %s %s %s %s",n.ISBN,n.book,n.author,n.edition,n.press,n.year)

printf("\n")

}

}

printf("\n")

printf("回到主菜单请按回车")

do

{

x=getchar()

}while(x!='\r')

if(x=='\r')

{

fclose(fp)

//main()子函数调用主函数,变成递归,不知程序怎么运行了;

system("cls")

return//返回主程序就可以了,主程序一直主菜单里运行

}

}

void Me(){

printf("作者姓名:0\n")

printf("作者学号:0\n")

}