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

news2025/1/9 14:38:28

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


目录

一.整体框架

二.书籍和书架

书籍(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




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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

项目点使用Redis作为缓存技术-自用

在spring boot项目中&#xff0c;使用缓存技术只需在项目中导入相关缓存技术的依赖包&#xff0c;并在启动类上使用EnableCaching开启缓存支持即可。 例如&#xff0c;使用Redis作为缓存技术&#xff0c;只需要导入Spring data Redis的maven坐标即可。 描述 使用Redis缓存高频数…

WSA子系统(一)

WSA子系统安装教程 Windows Subsystem for Android (WSA) 是微软推出的一项功能&#xff0c;它允许用户在 Windows 11 上运行 Android 应用程序。通过在 Windows 11 上引入 WSA&#xff0c;用户可以在其 PC 上轻松运行 Android 应用程序&#xff0c;从而扩展了用户的应用程序选…

RobotFramework之用例执行时添加命令行参数(十三)

学习目录 引言 标签tag 设置变量 随机执行顺序 设置监听器 输出日志目录和文件 引言 Robot Framework 提供了许多命令行选项&#xff0c;可用于控制测试用例的执行方式以及生成的输出。本节介绍一些常用的选项语法。 标签tag 之前文章我们介绍过&#xff0c;在测试套件…

Leetcode经典题目之“双指针交换元素“类题目

1 LC 27. 移除元素 class Solution {public int removeElement(int[] nums, int val) {int nnums.length;int s0;for(int i0;i<n;i){// 只有不等于目标值的时候才会进行交换&#xff0c;然后移动s指针if(nums[i]!val){swap(nums,i,s);}}return s;}void swap(int[]nums, int…

(数据结构)算法的时间复杂度

注意语句频度和时间复杂度的区别&#xff0c;语句频度是指语句执行的次数&#xff0c;不可以像时间复杂度一样近似次数和省略常数项

在线识别二维码工具

具体请前往&#xff1a;在线二维码识别解码工具--在线识别并解码二维码网址等内容

FlinkCDC数据实时同步Mysql到ES

考大家一个问题&#xff0c;如果想要把数据库的数据同步到别的地方,比如es,mongodb,大家会采用哪些方案呢&#xff1f; ::: 定时扫描同步&#xff1f; 实时日志同步? 定时同步是一个很好的方案&#xff0c;比较简单&#xff0c;但是如果对实时要求比较高的话&#xff0c;定…

HFSS螺旋线圈的设计与仿真

HFSS螺旋线圈的设计与仿真 HFSS中设计螺旋线圈的方法&#xff1a;参考文献&#xff1a; HFSS中设计螺旋线圈的方法&#xff1a; 打开软件Ansys Eletronics Desktop 2022 R1&#xff0c; 建立工程&#xff0c; File/New 插入HFSS设计&#xff0c; Project/Insert HFSS Design …

【Spring】之初识

未来的几周时间&#xff0c;大概率我会更新一下Spring家族的一些简单知识。而什么是Spring家族&#xff0c;好多同学还不是很清楚&#xff0c;我先来简单介绍一下吧&#xff1a; 所谓Spring家族&#xff0c;它其实就是一个框架&#xff0c;是基于Servlet再次进行封装的内容。为…

大模型的交互能力

摘要&#xff1a; 基础大模型显示出明显的潜力&#xff0c;可以改变AI系统的开发人员和用户体验&#xff1a;基础模型降低了原型设计和构建AI应用程序的难度阈值&#xff0c;因为它们在适应方面的样本效率&#xff0c;并提高了新用户交互的上限&#xff0c;因为它们的多模式和生…

笔记55:长短期记忆网络 LSTM

本地笔记地址&#xff1a;D:\work_file\DeepLearning_Learning\03_个人笔记\3.循环神经网络\第9章&#xff1a;动手学深度学习~现代循环神经网络 a a a a a a a a a

jbase打印完善

上一篇实现了粗略的打印元素绘制协议&#xff0c;并且写了打印示例和导出示例&#xff0c;趁着空隙时间完善一下打印。 首先元素构造函数默认初始化每个字段值 package LIS.Core.Dto;/*** 打印约定元素*/ public class PrintElement {/*** 元素类型*/public String PrintType…

OpenCV快速入门:图像形态学操作

文章目录 前言一、图像形态学基础1.1 背景介绍1.2 像素距离1.2.1 什么是像素距离&#xff1f;1.2.2 常见的像素距离度量方法1.2.3 计算像素距离的代码实现 1.3 图像连通性1.3.1 什么是图像连通性&#xff1f;1.3.2 连通类型1.3.3 连通组件标记1.3.4 连通性在图像处理中的应用 1…

2023.11.18 每日一题(AI自生成应用)【C++】【Python】【Java】【Go】 动态时间序列分析

目录 一、编程挑战&#xff1a;动态时间序列分析 实际应用&#xff1a; 实现提示&#xff1a; 二、实现 1. C 2. Python 3. JAVA 4. Go 一、编程挑战&#xff1a;动态时间序列分析 问题描述&#xff1a; 假设你是一名软件工程师&#xff0c;需要开发一个应用来分析和预…

内容运营策略:个性化推荐

一、推荐系统流程 典型的推荐系统包括3个部分&#xff0c;即召回层&#xff08; Recall )、排序层&#xff08; Rank )和重排层&#xff08; ReRank )。 1&#xff0e;召回层&#xff08; Recall ) 召回层主要是从全量库中首先获取用户可能感兴趣的候选集&#xff0c;是推荐系…

「Tech初见」对epoll的理解

一、Motivation 通常&#xff0c;操作系统会为每个进程划分一个时间片的&#xff0c;在这个时间片内进程可以合法占有 cpu 进行一些计算任务。并当时间片结束后自动退回至就绪状态待命&#xff0c;等待下一次的调度 但是&#xff0c;有一种情况会使进程提前&#xff08;时间片…

cad提示由于找不到mfc140u.dll,无法继续执行代码怎么修复

在Windows操作系统中&#xff0c;mfc140u.dll是一个重要的文件&#xff0c;很多软件运行都需要它&#xff0c;它属于Microsoft Visual C库的一部分。许多基于C的开发项目都依赖于这个文件&#xff0c;如果在使用过程中出现丢失现象&#xff0c;可能导致相关软件或游戏无法正常运…

洛谷 P1064 [NOIP2006 提高组] 金明的预算方案 python解析

P1064 [NOIP2006 提高组] 金明的预算方案 时间&#xff1a;2023.11.19 题目地址&#xff1a;[NOIP2006 提高组] 金明的预算方案 题目分析 动态规划的0-1背包&#xff0c;采用动态数组。如果不了解的话&#xff0c;可以先看看这个背包DP。 这个是0-1背包的标准状态转移方程 f…

SOME/IP 协议介绍(六)接口设计的兼容性规则

接口设计的兼容性规则&#xff08;信息性&#xff09; 对于所有序列化格式而言&#xff0c;向较新的服务接口的迁移有一定的限制。使用一组兼容性规则&#xff0c;SOME / IP允许服务接口的演进。可以以非破坏性的方式进行以下添加和增强&#xff1a; • 向服务中添加新方法 …

【C++】【Opencv】cv::Canny()边缘检测函数详解和示例

Canny边缘检测是一种流行的边缘检测算法&#xff0c;由John F. Canny在1986年开发。它是一种多阶段过程&#xff0c;包括噪声滤波、计算图像强度的梯度、非最大值抑制以及双阈值检测。本文通过函数原型解读和示例对cv::Canny()函数进行详解&#xff0c;以帮助大家理解和使用。 …