图书管理系统(图文详解,附源码)

前言:本文旨在用面向对象的思想编程实现图书管理系统,功能包括增删查找,完整源码放在文末,大家有需自取


目录

一.整体框架

二.书籍和书架

书籍(Book)

书架(BookRack)

三.对书籍的相关操作

操作接口(IOperation)

新增图书(AddOperation)

借阅图书(BorrowOperation)

删除图书(DeleteOperation)

查找图书(FindOperation)

归还图书(ReturnOperation)

展示图书(ShowOperation)

退出系统(ExitOperation)

四.用户部分

用户抽象类(User)

管理员类(Administrator)

普通用户(NormalUser)

五.main方法(Test)

完整代码:


一.整体框架

我们采取面向对象的编程思想,将整个图书管理系统抽象出多个对象,然后通过各个对象之间的交互来完成我们的整体设计需求

我们整体的设计框架如下图:

我们通过抽取他们的共性做出以下设计: 

  • 我们的书籍放在书架上,因此他们处于同一个包内
  • 我们的增删查改的操作都是属于用户对图书的操作,因此他们处于同一个包内,这样也更方便不同用户来调用这些操作
  • 用户分为普通用户和管理员用户,他们都是对于图书管理系统的直接操作者,因此他们处于同一个包中

对应我们上述结构图我们如下设计:


二.书籍和书架

书籍(Book)

我们应该提供书籍的相关信息:

  • 书名
  • 作者
  • 价格
  • 书籍类型
  • 借阅状态

为了体现面向对象的封装特性,我们将这些字段信息设为 private 然后再设置一些 public 的方法以供给其他对象访问

package BookRack;

//书籍
public class Book {
    private String name;//书名
    private String author;//作者名
    private int price;//价格
    private String type;//书的类型
    private boolean isBorrowed;//是否已经被借出
    
    public String getName() {
        return name;
    }
    
    public void setName(String name) {
        this.name = name;
    }
    
    public String getAuthor() {
        return author;
    }
    
    public void setAuthor(String author) {
        this.author = author;
    }
    
    public int getPrice() {
        return price;
    }
    
    public void setPrice(int price) {
        this.price = price;
    }
    
    public String getType() {
        return type;
    }
    
    public void setType(String type) {
        this.type = type;
    }
    
    public boolean isBorrowed() {
        return isBorrowed;
    }
    
    public void setBorrowed(boolean borrowed) {
        isBorrowed = borrowed;
    }
    
    //构造方法初始化
    public Book(String name, String author, int price, String type) {
        //图书默认没有借出,所以isBorrowed默认false,不需要初始化
        this.name = name;
        this.author = author;
        this.price = price;
        this.type = type;
    }
    
    //方便我们打印整个书籍的全部信息
    @Override
    public String toString() {
        return "Book.Book{" +
                "name='" + name + '\'' +
                ", author='" + author + '\'' +
                ", price=" + price +
                ", type='" + type + '\'' +
                '}';
    }
}

书架(BookRack)

书架主要是用来存放书籍的,因此我们只需要提供以下俩个信息就可以:

  • 已经存放的书籍,也就是一个书籍数组,数组中每一个元素都是一本书籍对象
  • 已经存放的书籍的数量

为了体现面向对象封装性,我们还是将这些字段信息设为 private 然后再设置一些 public 的方法以供给其他对象访问,在这里我们实现再书架上存放三本书,因此在构造方法中做出相应的初始化

package BookRack;

//书架
public class BookRack {
    private Book[] books;//存放的所有的书
    private int uesdSize;//书架上已经放的书的数量
    
    public BookRack() {
        this.books = new Book[10];//默认书架可以放10本书
        this.books[0] = new Book("三国演义","罗贯中",20,"小说");
        this.books[1] = new Book("西游记","吴承恩",9,"小说");
        this.books[2] = new Book("红楼梦","曹雪芹",19,"小说");
        this.uesdSize = 3;//默认书架上有3本书
    }
    //拿到某个位置的书籍
    public Book getBooks(int pos) {
        return books[pos];
    }
    //设置某个位置的书籍
    public void setBooks(Book book,int pos) {
        books[pos] = book;
    }
    
    public int getUesdSize() {
        return uesdSize;
    }
    
    public void setUesdSize(int uesdSize) {
        this.uesdSize = uesdSize;
    }
}

三.对书籍的相关操作

操作接口(IOperation)

所有的操作都是要对于书架进行操作的,所以我们在这里提供一个接口供不同操作来实现,并且给他们传入书架类的参数

package Operation;

import BookRack.BookRack;

//操作接口
public interface IOperation {
    //我们的任何增删查改的操作都是对于书架进行操作的,所以传入的参数是书架类
    void work(BookRack bookRack);
}

新增图书(AddOperation)

首先,我们需要让用户输入想要添加的图书的相关信息,然后我们为用户输入的图书新建一个对象,接下来就是合法性判断,我们拿刚才新建的图书对象和书架上的每一个图书对象进行遍历对比,如果没有重复的图书就可以存入这本书,要存入这本书就调用刚才书架类中提供的方法setBooks,在新加一本书籍后,对应的书籍数量也得增加,也就是调用setUesdSize方法来增加书籍的数量

package Operation;

import BookRack.BookRack;
import BookRack.Book;
import java.util.Scanner;

public class AddOperation implements IOperation{
    @Override
    public void work(BookRack bookRack) {
        System.out.println("新增图书操作进行中......");
        Scanner scanner = new Scanner(System.in);
        
        System.out.println("请输入您要添加的图书的书名:");
        String name = scanner.nextLine();
        System.out.println("请输入您要添加的图书的作者名:");
        String author = scanner.nextLine();
        System.out.println("请输入您要添加的图书的价格:");
        int price = scanner.nextInt();
        System.out.println("请输入您要添加的图书的类型:");
        String type = scanner.nextLine();
        
        //因为我们的每一个书都是一个对象,书架是一个对象数组,我们新加图书的时候就应该新实例化一个对象
        Book book = new Book(name,author,price,type);
        
        //合法性判断
        int usedSize = bookRack.getUesdSize();//拿到当前书架内放了书的数量
        for (int i = 0; i < usedSize; i++) {
            //遍历书架中的图书挨个对比名字是否相同
            Book tempbook = bookRack.getBooks(i);
            if (tempbook.getName().equals(name)) {
                System.out.println("不能重复添加同一本书,请重试!");
                return;
            }
        }
        
        //可以添加新的图书
        bookRack.setBooks(book,usedSize);
        bookRack.setUesdSize(usedSize+1);
    }
}

借阅图书(BorrowOperation)

首先,我们需要让用户输入想要添加的图书的相关信息,然后我们为用户输入的图书新建一个对象,然后我们挨个遍历书架上的书,如果有,那就可以借,将书籍的借阅状态改为true就可以,如果没有就告诉用户没有此书,无法借阅

package Operation;

import BookRack.BookRack;
import BookRack.Book;
import java.util.Scanner;

public class BorrowOperation implements IOperation{
    @Override
    public void work(BookRack bookRack) {
        System.out.println("借阅图书操作进行中......");
        System.out.println("请输入您想借阅的书名:");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        
        //遍历整个书架
        for (int i = 0; i < bookRack.getUesdSize(); i++) {
            
            Book temp = bookRack.getBooks(i);
            if (name.equals(temp.getName())) {
                temp.setBorrowed(true);
                System.out.println("借阅成功!");
                return;
            }
        }
        System.out.println("没有查询到您想要借阅的图书,请重新尝试!");
    }
}

删除图书(DeleteOperation)

首先,要删除图书的第一步应该是先找到这本书,因此我们像刚才借阅图书一样,先遍历整个书架找到这本书,然后记录这本书的位置,之后再利用书架提供的 setBooks 方法来存放这本书,如果没有找到那就告诉用户并且退出这个操作

package Operation;

import BookRack.BookRack;
import BookRack.Book;
import java.util.Scanner;

public class DeleteOperation implements IOperation{
    @Override
    public void work(BookRack bookRack) {
        System.out.println("删除图书操作进行中......");
        System.out.println("请输入您想删除的书名:");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        
        //要删除的前提是先找到这本书
        int uesdSize = bookRack.getUesdSize();
        int flag = -1;
        int i = 0;
        for (; i < uesdSize; i++) {
            Book tempbook = bookRack.getBooks(i);
            if (tempbook.getName().equals(name)) {
                //找到这本书
                flag = i;
                break;
            }
        }
        if (i >= uesdSize) {
            System.out.println("查无此书,无法删除");
            return;
        }
        //存在这本书,进行删除,也就是将书架中的书从后向前依次覆盖
        for (int j = flag; j < uesdSize; j++) {
            Book tempbook = bookRack.getBooks(j+1);//拿到 j+1 位置的书
            bookRack.setBooks(tempbook,j);//和 j 位置的书交换
        }
        bookRack.setBooks(null,uesdSize-1);//将最后一个位置的图书置为空
        bookRack.setUesdSize(uesdSize-1);//图书数量减一
        System.out.println("删除成功!");
    }
}

查找图书(FindOperation)

查找图书就非常简单了,我们在刚才的删除图书操作中相当于已经完成了这部分操作了

package Operation;

import BookRack.BookRack;
import BookRack.Book;
import java.util.Scanner;

public class FindOperation implements IOperation{
    @Override
    public void work(BookRack bookRack) {
        System.out.println("查找图书操作进行中......");
        System.out.println("请输入您想查找的书名:");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        int usedSize = bookRack.getUesdSize();
        
        for (int i = 0; i < bookRack.getUesdSize(); i++) {
            Book temp = bookRack.getBooks(i);
            if (name.equals(temp.getName())) {
                System.out.println("存在这本书,信息如下:");
                System.out.println(temp);
                return;
            }
        }
        System.out.println("没有你要找的这本书,书名为:"+ name);
    }
}

归还图书(ReturnOperation)

和我们的借阅图书操作相同,唯一不同的就是这里是将图书的借阅状态改为 false 

package Operation;

import BookRack.BookRack;
import BookRack.Book;
import java.util.Scanner;

public class ReturnOperation implements IOperation{
    @Override
    public void work(BookRack bookRack) {
        System.out.println("借阅图书操作进行中......");
        System.out.println("请输入您想借阅的书名:");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        
        //遍历整个书架
        for (int i = 0; i < bookRack.getUesdSize(); i++) {
            Book tempbook = bookRack.getBooks(i);
            if (name.equals(tempbook.getName())) {
                tempbook.setBorrowed(false);
                System.out.println("归还成功!");
                return;
            }
        }
        System.out.println("没有你要归还的图书:"+name);
    }
}

展示图书(ShowOperation)

遍历整个书架,然后挨个打印输出图书信息就可以了

package Operation;
import BookRack.Book;
import BookRack.BookRack;

public class ShowOperation implements IOperation{
    @Override
    public void work(BookRack bookRack) {
        System.out.println("图书列表如下:");
        for (int i = 0; i < bookRack.getUesdSize(); i++) {
            Book tempbook = bookRack.getBooks(i);
            System.out.println(tempbook);
        }
    }
}

退出系统(ExitOperation)

我们这里直接使用 exit 来结束整个程序就可以

package Operation;

import BookRack.BookRack;

public class ExitOperation implements IOperation{
    @Override
    public void work(BookRack bookRack) {
        System.out.println("退出系统...");
        System.exit(0);
    }
}

四.用户部分

用户抽象类(User)

在普通用户和管理员用户中存在许多共性,因此我们这里设置一个抽象类供普通用户和管理员用户来继承使用

package Person;

import BookRack.BookRack;
import Operation.IOperation;

public abstract class User {
    protected String name;//姓名
    protected IOperation[] iOperations;//操作接口数组
    public abstract int menu();//菜单
    
    public User(String name) {
        this.name = name;
    }
    //供用户来选择操作,调用操作接口
    public void doOperation(int choice, BookRack bookRack) {
        IOperation ioperation = iOperations[choice];
        ioperation.work(bookRack);
    }
}

管理员类(Administrator)

我们设置管理用户的菜单,再对应着菜单设置接口类型的数组的具体操作

package Person;

import Operation.*;

import java.util.Scanner;

public class Administrator extends User {
    public Administrator(String name) {
        super(name);
        iOperations = new IOperation[]{
                new ExitOperation(),
                new FindOperation(),
                new AddOperation(),
                new DeleteOperation(),
                new ShowOperation()
        };
    }

    public int menu() {
        System.out.println("********管理员菜单********");
        System.out.println("1.查找图书");
        System.out.println("2.新增图书");
        System.out.println("3.删除图书");
        System.out.println("4.显示图书");
        System.out.println("0.退出系统");
        System.out.println("************************");
        System.out.println("请输入你的操作:");
        //通过输入来调用对用的功能
        Scanner scanner = new Scanner(System.in);
        int choice = scanner.nextInt();
        return choice;
    }
}

普通用户(NormalUser)

和管理员用户的设置一样,我们对应的存放接口数组中的操作就可以

package Person;

import Operation.*;
import Person.User;

import java.util.Scanner;

public class NormalUser extends User {
    public NormalUser(String name) {
        super(name);
        iOperations = new IOperation[]{
                new ExitOperation(),
                new FindOperation(),
                new BorrowOperation(),
                new ReturnOperation()
        };
    }
    public int menu() {
        System.out.println("********普通用户菜单********");
        System.out.println("1.查找图书");
        System.out.println("2.借阅图书");
        System.out.println("3.归还图书");
        System.out.println("0.退出系统");
        System.out.println("***************************");
        System.out.println("请输入你的操作:");
        //通过输入来调用对用的功能
        Scanner scanner = new Scanner(System.in);
        int choice = scanner.nextInt();
        return choice;
    }
}

五.main方法(Test)

我们再这里设置一个登录程序,登录管理员我们就新建一个管理员对象,登录普通用户我们就新建一个普通用户对象,然后我们根据用户的输入来调用我们菜单中的选项,也就是我们刚才设置的接口操作数组中的操作

import BookRack.BookRack;
import Person.Administrator;
import Person.NormalUser;
import Person.User;

import java.util.Scanner;

public class Test {
    public static User login() {
        System.out.println("请输入您的名字:");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        
        System.out.println("请确认您的身份:");
        System.out.println("1.管理员");
        System.out.println("2.普通用户");
        int choice = scanner.nextInt();
        if (choice == 1) {
            return new Administrator(name);
        }else{
            return new NormalUser(name);
        }
    }
    
    public static void main(String[] args) {
        BookRack bookRack = new BookRack();
        User user = login();//向上转型
        while (true) {
            int choice = user.menu();
            user.doOperation(choice, bookRack);
        }
    }
}

完整代码:

按照笔者这里对应的包和类的设置进行操作就可以,每一个类和包在上文中都完整给出来了

如果觉得麻烦的话,笔者这里给出对应的码云,大家有需要自取

 LibrarySystem · 鹿鸣/JavaSE




 本次的分享就到此为止了,希望我的分享能给您带来帮助,也欢迎大家三连支持,你们的点赞就是博主更新最大的动力!如有不同意见,欢迎评论区积极讨论交流,让我们一起学习进步!有相关问题也可以私信博主,评论区和私信都会认真查看的,我们下次再见! 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/166700.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据结构【DS】栈

共享栈 共享栈的目的是什么&#xff1f; 目的:有效利用存储空间。 共享栈的存取数据时间复杂度为&#xff1f; 存取数据时间复杂度为O(1) 共享栈如何判空&#xff1f;如何判满&#xff1f; 两个栈的栈顶指针都指向栈顶元素&#xff0c;&#x1d461;&#x1d45c;&#x1d45d;…

【电路笔记】-欧姆定律

欧姆定律 文章目录 欧姆定律1、概述2、AC电路的等效性2.1 输入电阻2.2 输入电感2.3 输入电容 3、欧姆定律的局部形式3.1 介绍和定义3.2 德鲁德模型(Drude Model)3.3 局部形式表达式 4、电阻和宏观欧姆定律5、总结 电流、电压和电阻之间的基本关系被称为欧姆定律&#xff0c;可能…

C/C++高精度

个人主页&#xff1a;仍有未知等待探索_C语言疑难,数据结构,小项目-CSDN博客 专题分栏&#xff1a;算法_仍有未知等待探索的博客-CSDN博客 为什么需要高精度算法&#xff1f; 由于c不能进行位数过高的数据运算&#xff0c;所以要通过模拟数组来进行运算&#xff0c;首先是加法。…

参考文献格式

目录 期刊会议预印本&#xff08;如arxiv&#xff09; 期刊 找不到页码可以在文献中查看bibtex格式&#xff0c;其中有 外文期刊可在web of science中查找卷号、期号和所在页数&#xff1a; [1] ZHANG F, HU Z Q, FU Y K, et al. A New Identification Method for Surface …

【洛谷算法题】P5713-洛谷团队系统【入门2分支结构】

&#x1f468;‍&#x1f4bb;博客主页&#xff1a;花无缺 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! 本文由 花无缺 原创 收录于专栏 【洛谷算法题】 文章目录 【洛谷算法题】P5713-洛谷团队系统【入门2分支结构】&#x1f30f;题目描述&#x1f30f;输入格…

【SpringBoot3+Vue3】四【基础篇】-前端(vue基础)

目录 一、项目前置知识 二、使用vscode创建 三、vue介绍 四、局部使用vue 1、快速入门 1.1 需求 1.2 准备工作 1.3 操作 1.3.1 创建html 1.3.2 创建初始html代码 1.3.3 参照官网import vue 1.3.4 创建vue应用实例 1.3.5 准备div 1.3.6 准备用户数据 1.3.7 通过…

《许犁庭与柔性世界》第十六章 五大势力

“咱们伊拉斯蒂克学院的学生&#xff0c;大致分为五类&#xff0c;分别对应着弹性之城的五大势力。” “唔~” “第一类是极少数贵族家庭的孩子。他们背后是城主&#xff0c;秘书长与各大部长们&#xff0c;属于令老师们头疼&#xff0c;连院长都不敢管的角色。” “唔~” “第…

酷开会员丨酷开系统让居家K歌变得更简单!

音乐到底有着怎样的力量呢&#xff1f;一般的健身运动大多活动四肢和肌肉&#xff0c;而唱歌却能能按摩到内脏&#xff0c;促进脏腑健康。唱歌时&#xff0c;吸气与呼气间&#xff0c;横膈肌大幅度、频繁地上下移动&#xff0c;使胸腔、腹腔产生振动&#xff0c;这种震荡作用可…

[原创]解决老款AMD CPU在Win10/Win11无故重启的问题.

[简介] 常用网名: 猪头三 出生日期: 1981.XX.XXQQ: 643439947 个人网站: 80x86汇编小站 编程生涯: 2001年~至今[共22年] 职业生涯: 20年 开发语言: C/C、80x86ASM、PHP、Perl、Objective-C、Object Pascal、C#、Python 开发工具: Visual Studio、Delphi、XCode、Eclipse、C Bui…

信息中心网络提出的背景、研究现状及研究内容

信息中心网络什么时候提出的&#xff1f;未来发展前景&#xff1f;有什么著名实验室在做&#xff1f; 1、提出背景&#xff1a; 互联网产生于上世纪60年代&#xff1a; &#xff08;1&#xff09;网络设备数量呈指数性增长 截至2022年底全球范围内预计将有超过280亿台终端设…

汽车标定技术--A2L格式分析

目录 1.A2L由来 2.A2L格式 2.1 PROJECT 2.2 MODULE中包含的内容 3. INCA和CANape兼容吗&#xff1f; 最近有朋友用Vector ASAP2Editor编译的A2L文件在INCA7.4中无法识别&#xff0c;我记得以前做的时候是可以识别的&#xff0c;难不成最近有什么变动吗&#xff1f;出于好…

MySQL 的执行原理(五)

5.6 再深入查询优化 5.6.1. 全局考虑性能优化 5.6.3.1. 为什么查询速度会慢 在尝试编写快速的查询之前&#xff0c;需要清楚一点&#xff0c;真正重要是响应时间。如果把查询看作是一个任务&#xff0c;那么它由一系列子任务组成&#xff0c;每个子任务都会消耗一定的时间。…

Google codelab WebGPU入门教程源码<3> - 绘制网格(源码)

对应的教程文章: https://codelabs.developers.google.com/your-first-webgpu-app?hlzh-cn#4 对应的源码执行效果: 对应的教程源码: 此处源码和教程本身提供的部分代码可能存在一点差异。 class Color4 {r: number;g: number;b: number;a: number;constructor(pr 1.0, p…

【实用技巧】更改ArduinoIDE默认库文件位置,解放系统盘,将Arduino15中的库文件移动到其他磁盘

本文主要介绍更改Arduino IDE &#xff08;含2.0以上版本&#xff09;默认库文件位置的方法。 原创文章&#xff0c;转载请注明出处&#xff1a; 【实用技巧】更改ArduinoIDE默认库文件位置&#xff0c;解放C盘&#xff0c;将Arduino15中的库文件移动到其他磁盘-CSDN博客文章浏…

基于RK3588的8k多屏异显安卓智能网络机顶盒

采用RK3588芯片方案的8K网络机顶盒&#xff0c;搭载纯净的安卓12操作系统&#xff0c;支持Ubuntu和Debian系统容拓展。主要面向外贸市场。此款机顶盒自带两个HDMI输出接口&#xff0c;一个HDMI输入接口&#xff0c;内置双频WiFi6无线模块&#xff0c;支持千兆以太网和USB接口。…

SPASS-曲线估计

基本概念 曲线估计&#xff08;曲线拟合、曲线回归&#xff09;则是研究两变量间非线性关系的一种方法&#xff0c;选定一种用方程表达的曲线&#xff0c;使得实际数据与理论数据之间的差异尽可能地小。如果曲线选择得好&#xff0c;那么可以揭示因变量与自变量的内在关系&…

java拼图小游戏

第一步是创建项目 项目名自拟 第二部创建个包名 来规范class 然后是创建类 创建一个代码类 和一个运行类 代码如下&#xff1a; package heima;import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.KeyEvent; import jav…

Flutter笔记:缩放手势

Flutter笔记 缩放手势 作者&#xff1a;李俊才 &#xff08;jcLee95&#xff09;&#xff1a;https://blog.csdn.net/qq_28550263 邮箱 &#xff1a;291148484163.com 本文地址&#xff1a;https://blog.csdn.net/qq_28550263/article/details/134485138 目 录 1. 概述2. 缩放手…

数据结构【DS】图的应用

图的连通性问题 最少边数 最多边数 无向图非连通 &#x1d48e;&#x1d7ce; &#x1d48e;&#x1d48f;−&#x1d7d0;∗(&#x1d48f;−&#x1d7cf;)/&#x1d7d0; 无向图连通 &#x1d48e;&#x1d48f;−&#x1d7cf; &#x1d48e;&#x1d48f;∗(&#…