实战项目——用Java实现图书管理系统
前言
首先既然是管理系统,那咱们就要实现以下这几个功能了-->


分析
1.首先是用户分为两种,一个是管理员,另一个是普通用户,既如此,可以定义一个用户类(user),在定义管理员类,普通用户类去继承。
2.还有就是书这个对象,首先是书的属性(书名,作者,价格,种类,是否借出),这可以定义成一个Book类,其实就是存储这些新的容器-- 书架类(Booklist)
3.最后这些功能,仔细观察,由于身份不同,操作也不同,一般来讲,我们可以把对应的方法放到对应的类当中去,但这次用接口也可以巧妙实现
分模块进行编码
book包
为了方便查看,就把这两个类放进一个book的包里。

Book类
package book;public class Book {private String name;//姓名private String author;//作者private int price;//价格private String type;//类型private boolean isLead;//是否被借出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 isLead() {return isLead;}public void setLead(boolean lead) {isLead = lead;}public Book(String name, String author, int price, String type) {this.name = name;this.author = author;this.price = price;this.type = type;}@Overridepublic String toString() {return "book{" +"name='" + name + '\'' +", author='" + author + '\'' +", price=" + price +", type='" + type + '\'' +(isLead==true ? "已借出":" 未借出")+//", isLead=" + isLead +'}';}
}
在这个类中,把书的属性成员定义为私有,通过公共成员函数去访问和调用,最后重写toString 方法来表示输出。
Booklist类
package book;public class Booklist {private Book[] books =new Book[10];private int usedSize;//记录存放次数public Booklist() {books[0]=new Book("红楼梦","曹雪芹",25,"小说");books[1]=new Book("月华","小明",27,"小说");books[2]=new Book("多少","小花",20,"小说");this.usedSize=3;}public boolean isFull(){//判断书架是否满if(this.usedSize==books.length){return true;}else{return false;}}public int getUsedSize() {//获取存了多少本书return usedSize;}//设定书的数量public void setUsedSize(int usedSize) {this.usedSize = usedSize;}public Book getBooks(int pos) {//获取书架中pos位置书的信息return books[pos];}//把新增的图书信息放到pos位置上去public void setBooks(int pos,Book book) {books[pos]=book;}}
在这个书架类中,定义一个容量为10的对象数组来存储书的信息,事先存入三本书,同时usedSize 用来记录书的数量,而后面的方法则与实现对应的功能相关。且往后细看~接口类
package operater;import book.Booklist;public interface IOperation {void work(Booklist booklist);
}
在这个接口里定义了一个方法 work,该方法需要一个 Booklist 对象作为参数。任何实现 IOperation 接口的类都必须提供 work 方法的具体实现。这样一来就方便不同的用户进行操作了。
User包

user类
package User;import book.Booklist;
import operater.IOperation;// 定义一个抽象类 user,表示用户
public abstract class user {protected String name;// 构造方法,初始化用户名称public user(String name) {this.name = name;}//抽象方法,显示菜单并返回用户选择的操作编号public abstract int menu();//存储用户可以执行的操作数组protected IOperation[] iOperations;//根据用户选择执行相应的操作public void DoIoperation(int choice, Booklist booklist) {//调用用户选择的操作的 work 方法this.iOperations[choice].work(booklist);}
}因为身份不同,所以显示的菜单也不同,所以便声明一个抽象方法 menu(),同时定义一个操作数组,来进行对应的操作,如图

普通用户类(Normaluser)
package User;import operater.*;import java.util.Scanner;public class Normaluser extends user{//重写构造方法public Normaluser(String name) {super(name);//根据不同的输入进行对应的操作,已经方法按顺序放入数组,所以位置别搞错了this.iOperations=new IOperation[]{new Exit(),new Find(),new Borrow(),new Return()};}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("请输入你的操作:");Scanner scanner=new Scanner(System.in);int choice=scanner.nextInt();return choice;}
}
管理员类(Adminuser)
package User;import operater.*;import java.util.Scanner;public class Adminuser extends user{public Adminuser(String name) {super(name);this.iOperations=new IOperation[]{new Exit(),new Find(),new Add(),new Del(),new Show()};}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;}}operater包

Add类
package operater;import book.Book;
import book.Booklist;import java.util.Scanner;public class Add implements IOperation{public void work(Booklist booklist1){if(booklist1.isFull()){System.out.println("书架已满,不能新增!");}System.out.println("新增图书");System.out.println("请输入你要新增的图书的书名:");Scanner scanner=new Scanner(System.in);String bookName=scanner.nextLine();System.out.println("请输入你要新增的图书的作者:");String bookAuthor= scanner.nextLine();System.out.println("请输入你要新增的图书的价格:");int bookPrice = scanner.nextInt();scanner.nextLine();System.out.println("请输入你要新增的图书的类型:");String bookType = scanner.nextLine();Book book=new Book(bookName,bookAuthor,bookPrice,bookType);int currentSize=booklist1.getUsedSize();//新增图书的位置其实就是i本书的位置booklist1.setBooks(currentSize,book);booklist1.setUsedSize(currentSize+1);//书的数量+1System.out.println("成功添加图书");}
}
在新增图书的过程中,要分别添加书的属性才行,这时切不可直接用booklist,这时类对象,不是数组,创建一个对象数组book,之前Booklist中方法便派上用场, 看图就更好理解了~

Borrow类
package operater;import book.Book;
import book.Booklist;import java.util.Scanner;public class Borrow implements IOperation{public void work(Booklist booklist1){System.out.println("借阅图书");System.out.println("请输入你要借阅的书名:");Scanner scanner=new Scanner(System.in);String bookName=scanner.nextLine();//找到要借阅的书的下标for (int i=0; i < booklist1.getUsedSize() ; i++) {Book book1=booklist1.getBooks(i);if(book1.getName().equals(bookName)) {book1.setLead(true);System.out.println("借阅成功!");return;}}System.out.println("借阅失败!");}}借阅图书其实和删除差不多,只要改变状态为已借出。
Del类
package operater;import book.Book;
import book.Booklist;import java.util.Scanner;public class Del implements IOperation{public void work(Booklist booklist1){System.out.println("删除图书");System.out.println("请输入你要删除的书名:");Scanner scanner=new Scanner(System.in);String bookName=scanner.nextLine();int current=booklist1.getUsedSize();int pos=-1;int i = 0;//找到要删除的书的下标for (; i < current ; i++) {Book book1=booklist1.getBooks(i);if(bookName.equals(book1.getName())) {//System.out.println("找到了");pos=i;}}if(pos>=current){System.out.println("没有你要找的书!");}//开始删除for (int j = pos; j <current-1; j++) {// booklist1[j]=booklist1[j+1];不能直接用类对象Book book= booklist1.getBooks(j+1);booklist1.setBooks(j,book);}booklist1.setUsedSize(current-1);booklist1.setBooks(current-1,null);System.out.println("删除成功!");}
}
删除图书,首先要找到要删除图书的下标pos, 找到后要把后面的图书全都往前移动一格,从pos位置开始遍历,将j+1位置的图书存放到book对象中去,在调用setBook方法传给j位置。后面的位置设置成null,同时更新书的数量~

Exit类
package operater;import book.Booklist;public class Exit implements IOperation{public void work(Booklist booklist1){System.out.println("退出系统");//避免退出系统后内存被回收int current=booklist1.getUsedSize();for (int i = 0; i < current ; i++) {booklist1.setBooks(i,null);}System.exit(0);}
}
直接调用系统的退出即可~
Find类
package operater;import book.Book;
import book.Booklist;import java.util.Scanner;public class Find implements IOperation{public void work(Booklist booklist1){System.out.println("查找图书");System.out.println("请输入你要查找的书名:");Scanner scanner=new Scanner(System.in);String bookName=scanner.nextLine();int current=booklist1.getUsedSize();for (int i = 0; i < current ; i++) {Book book1=booklist1.getBooks(i);if(bookName.equals(book1.getName())) {System.out.println("找到了");System.out.println(book1);return;}}System.out.println("没找到这本书!");}
}
Return类
package operater;import book.Book;
import book.Booklist;
import java.util.Scanner;public class Return implements IOperation{public void work(Booklist booklist1){System.out.println("归还图书....");System.out.println("请输入你要归还的书名:");Scanner scanner=new Scanner(System.in);String bookName=scanner.nextLine();//找到要归还的书的下标for (int i=0; i < booklist1.getUsedSize() ; i++) {Book book1=booklist1.getBooks(i);if(book1.getName().equals(bookName)) {book1.setLead(false);System.out.println("归还成功!");return;}}System.out.println("归还失败!");}}Show类
package operater;import book.Book;
import book.Booklist;public class Show implements IOperation{public void work(Booklist booklist1){System.out.println("展示图书");int current=booklist1.getUsedSize();for (int i = 0; i < current ; i++) {Book book1=booklist1.getBooks(i);System.out.println(book1);}}
}
这三个其实是大同小异的,都是循环遍历再进行对应的操作,这里主要记住的是String类型的数据比较不能直接用==,而是要用equals()方法。

看到最后,如果觉得文章写得还不错,希望可以给我点个小小的赞,您的支持是我更新的最大动力

相关文章:
 
实战项目——用Java实现图书管理系统
前言 首先既然是管理系统,那咱们就要实现以下这几个功能了--> 分析 1.首先是用户分为两种,一个是管理员,另一个是普通用户,既如此,可以定义一个用户类(user),在定义管理员类&am…...
 
利用DeepFlow解决APISIX故障诊断中的方向偏差问题
概要:随着APISIX作为IT应用系统入口的普及,其故障定位能力的不足导致了在业务故障诊断中,APISIX常常成为首要的“嫌疑对象”。这不仅导致了“兴师动众”式的资源投入,还可能使诊断方向“背道而驰”,从而导致业务故障“…...
sqlalchemy获取数据条数
1、sqlalchemy获取数据条数 在SQLAlchemy中,你可以使用count()函数来获取数据表中的记录条数。 from sqlalchemy import create_engine, MetaData, Table# 数据库连接字符串 DATABASE_URI = dialect+driver://username:password@host:port/database# 创建引擎 engine = crea…...
 
SpringBoot的自动配置核心原理及拓展点
Spring Boot 的核心原理几个关键点 约定优于配置: Spring Boot 遵循约定优于配置的理念,通过预定义的约定,大大简化了 Spring 应用程序的配置和部署。例如,它自动配置了许多常见的开发任务(如数据库连接、Web 服务器配…...
 
用随机森林算法进行的一次故障预测
本案例将带大家使用一份开源的S.M.A.R.T.数据集和机器学习中的随机森林算法,来训练一个硬盘故障预测模型,并测试效果。 实验目标 掌握使用机器学习方法训练模型的基本流程;掌握使用pandas做数据分析的基本方法;掌握使用scikit-l…...
 
24位DAC转换的FPGA设计及将其封装成自定义IP核的方法
在vivado设计中,为了方便的使用Block Desgin进行设计,可以使用vivado软件把自己编写的代码封装成IP核,封装后的IP核和原来的代码具有相同的功能。本文以实现24位DA转换(含并串转换,使用的数模转换器为CL4660)为例,介绍VIVADO封装IP核的方法及调用方法,以及DAC转换的详细…...
 
【大模型LLM面试合集】大语言模型基础_llm概念
1.llm概念 1.目前 主流的开源模型体系 有哪些? 目前主流的开源LLM(语言模型)模型体系包括以下几个: GPT(Generative Pre-trained Transformer)系列:由OpenAI发布的一系列基于Transformer架构…...
 
Qt时间日期处理与定时器使用总结
一、日期时间数据 1.QTime 用于存储和操作时间数据的类,其中包括小时(h)、分钟(m)、秒(s)、毫秒(ms)。函数定义如下: //注:秒(s)和毫秒(ms)有默认值0 QTime::QTime(int h, int m, int s 0, int ms 0) 若无须初始化时间数据,可…...
数据结构——Hash Map
1. Hash Map简介 Hash Map是一种基于键值对的数据结构,通过散列函数将键映射到存储位置,实现快速的数据查找和存储。它可以在常数时间内完成查找、插入和删除操作,因此在需要频繁进行这些操作时非常高效。 2. Hash Map的定义 散列表ÿ…...
 
剪画小程序:视频剪辑-视频播放倍数的调整与应用
在这个快节奏的时代,时间变得越来越宝贵,而视频倍数播放功能就像是我们的时间管理小助手,为我们的视频观看带来了极大的便利。你是否好奇它到底能在哪些地方发挥作用呢?让我们一起来看看! 只要使用小程序【剪画】的里…...
 
使用 Java Swing 和 XChart 创建多种图表
在现代应用程序开发中,数据可视化是一个关键部分。本文将介绍如何使用 Java Swing 和 XChart 库创建各种类型的图表。XChart 是一个轻量级的图表库,支持多种类型的图表,非常适合在 Java 应用中进行快速的图表绘制。 1、环境配置 在开始之前&…...
 
信息系统运维管理:实践与发展
信息系统运维管理:实践与发展 信息系统运维管理在现代企业中扮演着至关重要的角色,确保信息系统的高效、安全和稳定运行。本文结合《信息系统运维管理》文档内容,探讨了服务设计阶段、服务转换阶段、委托系统维护管理三个主要章节࿰…...
 
html+js+css登录注册界面
拥有向服务器发送登录或注册数据并接收返回数据的功能 点赞关注 界面 源代码 <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8"> <title>Login and Registration Form</title> <style> * …...
英伟达(NVIDIA)数据中心GPU介绍
英伟达(NVIDIA)数据中心GPU按性能由高到低排行: 1. NVIDIA H100 架构:Hopper 核心数量:18352 CUDA Cores, 1456 Tensor Cores 显存:80 GB HBM3 峰值性能: 单精度(FP32)…...
Leetcode 3202. Find the Maximum Length of Valid Subsequence II
Leetcode 3202. Find the Maximum Length of Valid Subsequence II 1. 解题思路2. 代码实现 题目链接:3202. Find the Maximum Length of Valid Subsequence II 1. 解题思路 这一题的话是上一题3201. Find the Maximum Length of Valid Subsequence I的升级版&am…...
通过Spring Boot结合实时流媒体技术对考试过程进行实时监控
本章将深入探讨考试系统中常见的复杂技术问题,并提供基于Spring Boot 3.x的解决方案。涵盖屏幕切换检测与防护、接打电话识别处理、行为监控摄像头使用、网络不稳定应对等,每篇文章详细剖析问题并提供实际案例与代码示例,帮助开发者应对挑战&…...
智能扫地机器人避障与防跌落问题解决方案
智能扫地机器人出现避障与防跌落问题时,可以通过以下几种方式来解决: 一、避障问题的解决方案 1.升级避障技术: ① 激光雷达避障:激光雷达通过发射和接收激光信号来判断与障碍物的距离,具有延迟低、效果稳定、准确度…...
 
德旺训练营称重问题
这是考小学的分治策略,小学的分治策略几乎都是分三组。本着这个策略,我们做看看。 第一次称重: 分三组,16,16,17,拿两个16称,得到A情况,一样重,那么假铜钱在那组17个里面。B情况不…...
数据决策系统详解
文章目录 数据决策系统的核心组成部分:1. **数据收集与整合**:2. **数据处理与分析**:3. **数据可视化**:4. **决策支持**: 数据决策系统的功能:决策类型:数据决策系统对企业的重要性࿱…...
JSON 简述与应用
1. JSON 简述 JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,常用于客户端与服务器之间的数据传递。它基于JavaScript对象表示法,但独立于语言,可以被多种编程语言解析和生成。 1.1 特点 轻量级&#…...
 
《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...
 
基于当前项目通过npm包形式暴露公共组件
1.package.sjon文件配置 其中xh-flowable就是暴露出去的npm包名 2.创建tpyes文件夹,并新增内容 3.创建package文件夹...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
 
从零实现STL哈希容器:unordered_map/unordered_set封装详解
本篇文章是对C学习的STL哈希容器自主实现部分的学习分享 希望也能为你带来些帮助~ 那咱们废话不多说,直接开始吧! 一、源码结构分析 1. SGISTL30实现剖析 // hash_set核心结构 template <class Value, class HashFcn, ...> class hash_set {ty…...
 
用docker来安装部署freeswitch记录
今天刚才测试一个callcenter的项目,所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...
 
【论文阅读28】-CNN-BiLSTM-Attention-(2024)
本文把滑坡位移序列拆开、筛优质因子,再用 CNN-BiLSTM-Attention 来动态预测每个子序列,最后重构出总位移,预测效果超越传统模型。 文章目录 1 引言2 方法2.1 位移时间序列加性模型2.2 变分模态分解 (VMD) 具体步骤2.3.1 样本熵(S…...
LeetCode - 199. 二叉树的右视图
题目 199. 二叉树的右视图 - 力扣(LeetCode) 思路 右视图是指从树的右侧看,对于每一层,只能看到该层最右边的节点。实现思路是: 使用深度优先搜索(DFS)按照"根-右-左"的顺序遍历树记录每个节点的深度对于…...
 
AI+无人机如何守护濒危物种?YOLOv8实现95%精准识别
【导读】 野生动物监测在理解和保护生态系统中发挥着至关重要的作用。然而,传统的野生动物观察方法往往耗时耗力、成本高昂且范围有限。无人机的出现为野生动物监测提供了有前景的替代方案,能够实现大范围覆盖并远程采集数据。尽管具备这些优势…...
 
Python训练营-Day26-函数专题1:函数定义与参数
题目1:计算圆的面积 任务: 编写一个名为 calculate_circle_area 的函数,该函数接收圆的半径 radius 作为参数,并返回圆的面积。圆的面积 π * radius (可以使用 math.pi 作为 π 的值)要求:函数接收一个位置参数 radi…...
 
goreplay
1.github地址 https://github.com/buger/goreplay 2.简单介绍 GoReplay 是一个开源的网络监控工具,可以记录用户的实时流量并将其用于镜像、负载测试、监控和详细分析。 3.出现背景 随着应用程序的增长,测试它所需的工作量也会呈指数级增长。GoRepl…...

