【JavaSE】javaSE练习项目——>《简易图书管理系统》

news2024/11/15 17:53:06

目录

前言:

1、项目实现要求

2、设计思路流程

 设计思路:

登录后菜单的实现效果:

3、代码实现(大体框架)

Main类

book包

Book类

BookList类

 user包

User类

AdminUser(管理员)类

NormalUser(普通用户)类

 4、业务逻辑(实现opera包中的类)

管理员的业务逻辑

显示图书(ShowOperation)

查找图书(FindOperation)

退出系统(ExitOperation)

增加图书(AddOperation)

 删除图书(DelOperation)

 普通用户的业务逻辑

借出图书(BrrowOperation)

归还图书(ReturnOperation)

5、最终实现效果

管理员:

 普通用户:


前言:

  • 这篇博客是在学习了一部分Java基础语法后的练习项目,通过这个练习对我们之前学过的数组、类和对象、抽象类和接口、继承、多态和封装等知识点进行巩固;也上我们初次了解一下面向对象编程的思想。
  • 面向对象编程,所有的操作都是通过对象和动作的交互完成的。

1、项目实现要求

图书管理系统,面向管理员和普通用户使用,对管理员开放的功能有:增加图书、删除图书、查找图书、显示图书、退出系统;对普通用户开放的功能有:查找图书、借阅图书、归还图书、退出系统。

2、设计思路流程

 设计思路:

  1. 根据上述图片将图书管理系统的对象可以抽取成为两类:用户和书
  2. 所以根据两类对象,分别创建user包和book包;分别在两个包当中创建相应的类。
  3. 用户类对象又可分为:管理员和普通用户;两类对象有存在共有属性,所以我们可以将他们共有的属性设置为一个User父类的抽象类。
  4. 书类对象又可分为:书和书架;书本身可以抽象为一个对象,书架用来对多本书进行管理,所以书架也可以抽象为一个对象,所以书和书架可以设置为两个类来实现,书架可以设置为一个数组。
  5. 当然对象设置完成之后,对象所要实现的功能我们可以再设置一个opera包,再opera包中设置一个接口IOperation,每个功能设置为一个类并且都实现接口IOperation。

登录后菜单的实现效果:

管理员身份:

 普通用户:


3、代码实现(大体框架)

Main类

import book.BookList;
import user.AdminUser;
import user.NormalUser;
import user.User;

import java.util.Scanner;

public class Main {
    //发生向上转型 User作为返回值类型
    public static User login(){//登录
        System.out.println("请输入你的姓名:");
        Scanner scanner =new Scanner(System.in);
        String name = scanner.nextLine();
        System.out.println("请输入你的身份:1->管理员  0->普通用户");
        int choice = scanner.nextInt();
        if(choice == 1){//选择用户
           return new AdminUser(name );//返回管理员对象
        }else{
           return new NormalUser(name );//返回普通用户对象
        }
    }
    public static void main(String[] args) {
        //准备数据
        BookList booklist = new BookList();
        //登录
        User user = login();//将login方法中返回的用户身份用user接收
        while (true){//用while来实现循环操作
            int choice = user.menu();//这里的choice用来接收AdminUser和NormalUser中返回的操作;
            user.doWork(choice, booklist);//这里的user代表的是AdminUser或者NormalUser
         }
    }
}

❗❗❗注意:在login方法中,返回值有两个,且是两种不同类型的对象,并且要传给User类型的user变量,所以不能设置为void,又因为返回的两个对象都是User的子类,所以用User作为返回值类型,并且在main方法中用User类型的user变量来接收,这里就发生了向上转型。

book包

Book类

package book;

public class Book {
    private String name;//书名
    private String author;//作者
    private int price;//价格
    private String type;//类型
    private boolean isBorrowed;//是否被借出
//写一个构造方法,但是isBorrowed 属性不因该写在构造方法中,因为在添加一本新书的时候,默认都是未借出的
    public Book(String name, String author, int price, String type) {
        this.name = name;
        this.author = author;
        this.price = price;
        this.type = type;
    }
    //这些属性都是以封装的形式出现的,所以都要提供公开的get和set方法
    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;
    }

    @Override
    public String toString() {
        return "Book{" +
                "name='" + name + '\'' +
                ", author='" + author + '\'' +
                ", price=" + price +
                ", type='" + type + '\'' +
               (isBorrowed == true?"已经被借出":"未被借出")+
                '}';
    }
}

BookList类

package book;

public class BookList {
    private static final int DEFAULT_SIZE = 10;
    private Book[] books = new Book[DEFAULT_SIZE ];
    //private Book[] books = new Book[10];
    private int usedSize;//记录当前books数组当中,有多少本书
    //通过构造方法给books这个书架中放书,并通过usedSize来记录书本个数。
    public BookList(){
        books[0] = new Book("三国演义","罗贯中",89,"小说");
        books[1] = new Book("西游记","吴承恩",78,"小说");
        books[2] = new Book("红楼梦","曹雪芹",49,"小说");
        this.usedSize = 3;
    }
    //因为books被private分装了,所以要提供一个get和set方法,
    public Book getBook(int pos){
        return this.books[pos];
        //返回这个books数组中pos下标的元素
    }
    //这个setBook方法被用于增添图书的功能中
    public void setBook(Book book){
        books[usedSize] = book;//这里写死了,给下标为usedSize的地方添加书,
    }
    //这个setBook方法被用于删除图书的功能中
    public void setBook(int pos,Book book){
        books[pos] = book;//表示给数组下标为pos的地方。放一本书
    }

    public int getUsedSize() {
        return usedSize;
    }

    public void setUsedSize(int usedSize) {
        this.usedSize = usedSize;
    }
}

 user包

User类

package user;

import book.BookList;
import opera.IOperation;

public abstract class User {
    protected String name;
    protected IOperation[] ioperations;//定义一个数组,用来存放相应用户的操作

    public User(String name) {
        this.name = name;
    }
    public abstract int menu();//在User类中实现menu方法没有意义,所以定义成为一个抽象方法。
    //doWork方法用来将AdminUser对象和NormalUSer对象中的choice选择的数组下标所对应的操作类调用起来
    public void doWork(int choice ,BookList booklist){
       this.ioperations[choice].work(booklist);
       //调用ioperation数组的choice下标所对应的操作对象中的work方法
    }
}

AdminUser(管理员)类


package user;
import opera.*;

import java.util.Scanner;

public class AdminUser extends User {
    public AdminUser(String name) {
        super(name);
        this.ioperations =new IOperation[]{//动态扩容
                new ExitOperation(),
                new FindOperation(),//这里new的操作对象,要和下边的菜单对应起来
                new AddOperation(),
                new DelOperation(),
                new ShowOperation(),

        };
    }
    public int  menu(){
        System.out.println("******************************");
        System.out.println("hello"+name+"欢迎来到图相互小练习");
        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();//这里的choice用来接收管理员用户在菜单中的选项
        return choice ;//这里返回的choice 被Main类中的choice接收
    }
}

NormalUser(普通用户)类

package user;

import opera.*;

import java.util.Scanner;

public class NormalUser extends User {
    public NormalUser(String name) {
        super(name);
        this.ioperations = new IOperation[]{
                new ExitOperation(),
                new FindOperation(),
                new BrrowOperation(),
                new ReturnOperation(),
        };
    }
    public int  menu(){
        System.out.println("******************************");
        System.out.println("hello"+name+"欢迎来到图相互小练习");
        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 ;
    }
}

 4、业务逻辑(实现opera包中的类)

管理员的业务逻辑

显示图书(ShowOperation)

package opera;

import book.Book;
import book.BookList;

public class ShowOperation implements IOperation{
    @Override
    public void work(BookList booklist) {
        System.out.println("显示图书!");
        int currentSize = booklist.getUsedSize();
        //通过currentSize变量来记录BookList对象中传过来的usedSize,
        //通过for循环将books数组中的所有元素遍历输出
        for (int i = 0; i < currentSize; i++) {
            //通过booklist调用BookList中的getBook方法,得到下标为i的books数组中的元素
           Book book = booklist.getBook(i);
          //输出这本书
            System.out.println(book);
        }
    }
}

查找图书(FindOperation)

package opera;

import book.Book;
import book.BookList;

import java.util.Scanner;

public class FindOperation implements IOperation {
    @Override
    public void work(BookList booklist) {
        System.out.println("查找图书!");
        System.out.println("请输入要查找的书名:");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        //通过currentSize变量来记录booklist引用调用getUsedSize传过来的usedSize
        int currentSize = booklist.getUsedSize();
        //通过for循环来遍历查找
        for (int i = 0; i < currentSize; i++) {
            //通过booklist调用BookList对象中的getBook方法,得到books数组下标为i元素
            Book book = booklist.getBook(i);
            //通过book引用来调用Book类中的getName方法,得到书名和查找的书名通过equals来比较。
            if (book.getName().equals(name)) {
                System.out.println("找到了这本书");
                System.out.println(book);
                //返回找到的这本书
                return;
                //这里直接return,我们假设没有重复的书名
            }
        }
        System.out.println(" 没有这本书!");
    }
}

退出系统(ExitOperation)

package opera;

import book.BookList;

public class ExitOperation implements IOperation {
    @Override
    public void work(BookList booklist) {
        System.out.println("退出系统!");
        System.exit(0);
        //正常系统推出显示是0,在这里设置为0.
    }
}

增加图书(AddOperation)

package opera;

import book.Book;
import book.BookList;

import java.util.Scanner;


public class AddOperation implements IOperation {
    @Override
    public void work(BookList booklist){
        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("输入类型:");
        String type = scanner.nextLine();
        System.out.println("输入价格:");
        int price = scanner.nextInt();


        //以上属性,可以构成书的对象
        Book book = new Book(name,author,price,type);
        //新增的这本书,是要放在books着这个数组中,所以不能存在相同的书,
        // 所以要通过遍历来查找有没有相同的书,如果没有则加入到books数组中,如果有则不用加入
        int currentSize = booklist.getUsedSize();
        for (int i = 0; i < currentSize; i++) {
            //通过booklist调用BookList中的getBook方法,得到下标为i的books数组中的元素
            Book tmp = booklist.getBook(i);
            if (tmp.getName().equals(name)) {
                System.out.println("已经存在这本书了,不能再放入了!");
                return;
            }
        }
        booklist.setBook(book);
        //修改usedSize
        booklist.setUsedSize(currentSize+1);
//这里通过setusedSize方法,将currentSize+1传到BookList书架对象中,
//对usedSize经行修改。currentSize通过BookList的get方法得到usedSize的值,
//所以currentSize+1,就表示对usedSize进行加1
    }

}

 画图理解思路:

 删除图书(DelOperation)

删除图书中的难点:

  • 挪动数据
  • 修改usedSize
  • 将最后一位置空(防止内存泄漏)
package opera;

import book.Book;
import book.BookList;

import java.util.Scanner;

public class DelOperation implements IOperation{
    @Override
    public void work(BookList booklist) {
        System.out.println("删除图书!");
        System.out.println("请输入要删除的图书:");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        int currentSize = booklist.getUsedSize();
       //设置index用来接收要删除的书籍的下标
        int index = -1;
        for (int i = 0; i < currentSize; i++) {
            Book tmp = booklist.getBook(i);
            if (tmp.getName().equals(name)){
                index = i;
                break;
            }
        }
        //挪动数据,currentSize-1表示的是数组中存在元素的最后一位
        for (int j = index; j < currentSize-1; j++) {
       //这里的j < currentSize-1不能改为j < currentSize或j <= currentSize-1
//这样在挪动数据的时候会发生数组越界
            Book book = booklist.getBook(j+1);
           //通过BookList的引用调用getBook这个方法,得到数组j+1下标的元素内容
            booklist.setBook(j,book );
//通过BookList的引用调用setBook方法,在Books这个数组指定的j下标的位置放一本书
        }
        //修改usedSize
        booklist.setUsedSize(currentSize-1);
        //因为删除的是对象,所以 把最后一个设置为null
        booklist.setBook(currentSize-1,null);
        System.out.println("删除成功!");

    }
}

画图理解:

 想要解决这个问题,在上述代码中通过修改usedSize,并且通过BookList的引用来调用setBook方法对最后一位赋一个空值。来解决删除最后一位的问题,usedSize-1,表示书架上的书被删除了一本。但是在 booklist.setBook(currentSize-1,null);这句代码中的意思是,在数组下标的最后一位,放null。

画图解释:


 普通用户的业务逻辑

借出图书(BrrowOperation)

package opera;

import book.Book;
import book.BookList;

import java.util.Scanner;

public class BrrowOperation implements IOperation {
    @Override
    public void work(BookList booklist) {
        System.out.println("借阅图书!");
        System.out.println("输入你要借阅的图书");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        int currentSize = booklist.getUsedSize();
        for (int i = 0; i < currentSize; i++) {
            Book book = booklist.getBook(i);
            //这里将输入的书名与书架中的书名进行比较并且查看是否被借出,
//因为isBorrowed属性在设置的时候为Boolean类型,所以用false来判断
            if(book.getName().equals(name)&&book.isBorrowed()==false){
                book.setBorrowed(true);
                System.out.println("借阅成功");
                return;
            }
        }
    }
}

归还图书(ReturnOperation)

package opera;

import book.Book;
import book.BookList;

import java.util.Scanner;

public class ReturnOperation implements IOperation {
    @Override
    public void work(BookList booklist) {
        System.out.println("归还图书!");
        System.out.println("输入你要归还的图书");
        Scanner scanner = new Scanner(System.in);
        String name = scanner.nextLine();
        int currentSize = booklist.getUsedSize();
        for (int i = 0; i < currentSize; i++) {
            Book book = booklist.getBook(i);
            //这里将输入的书名与书架中的书名进行比较并且查看是否被借出,因为isBorrowed属性在设置的时候为Boolean类型,所以用true来判断
            if(book.getName().equals(name)&&book.isBorrowed()==true){
                book.setBorrowed(false);
                System.out.println("归还成功");
                return;
            }
        }
    }
}

5、最终实现效果

管理员:

 普通用户:

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

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

相关文章

Hibernate Validator 使用详解

目录 Hibernate Validator的依赖 Hibernate Validator 支持注解 空与非空检查 Boolean值检查 日期检查 数值检查 其他 Hibernate-validator扩展约束 Hibernate Validator 校验 简单对象校验 嵌套对象校验 Hibernate Validator 分组校验 静态分组 动态分组 动态分…

【软件测试】刚入职后,快速适应新的工作需要做啥?

目录&#xff1a;导读前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09;前言 入职一家新公司后&a…

基于AD8226的环境光传感器电路

人们越来越多地认为环境光是一种能源,可用于驱动心率监控器、浴室灯具、远程天气传感器和其他低功耗器件。对于能量采集系统,最关键的是精确测量环境光的能力。本设计思路将描述一种简单的低成本电路,可以根据环境光的强度按一定比例提供电压。 所用传感器是一款光敏电阻(L…

spring framework 容器

org.springframework.beans 和 org.springframework.context 包是 Spring Framework 的 IoC 容器的基础。 这里需掌握两个体系结构&#xff0c;BeanFactory 和 ApplicationContext。 BeanFactory 主要接口&#xff0c;可分为三级&#xff1a; BeanFactory 是顶层容器&#xf…

Nacos 配置中心之主动拉取

客户端 客户端的配置有两种方式来维持,一是客户端主动拉取,而是客户端长轮询更新 配置文件的种类 1、本地配置文件: 本地就已经存在的配置文件 2、 本地缓存文件: 从服务端获取的保存在了本地 (本地生成了文件) 3、 cacheData 缓存数据: 内存中缓存的配置文件数据 客户端主动获…

【分享】学浪PC端登录分析及实现

本文所有教程及源码、软件仅为技术研究。不涉及计算机信息系统功能的删除、修改、增加、干扰,更不会影响计算机信息系统的正常运行。不得将代码用于非法用途,如侵立删!学浪PC端登录分析及实现 环境 win10Fiddlerchrome学浪PC端登录:aHR0cHM6Ly9zdHVkZW50LWFwaS5peWluY2Fpc2…

AE 动效制作和交付方案

在界面设计中&#xff0c;设计师利用动效让整个界面更加活泼&#xff0c;给界面元素带来生命力&#xff0c;解决功能上的问题&#xff0c;在更好地展示产品功能的基础上&#xff0c;凸显品牌的特色。而作为用户&#xff0c;动效增强了体验者的审美感受、情感需要&#xff0c;让…

德才论

目录 1015:德才论 输入格式&#xff1a; 输出格式&#xff1a; 输入样例&#xff1a; 输出样例&#xff1a; 思路: 1.技巧&#xff1a; 1.2结构体代码: 2.分类: 1.德才分类 1.2德才分类代码: 2.cmp函数 2.1 cmp函数代码: 3.超时问题(易错&#xff0c;算法要优化) 总代码…

神经网络优化

提升深度神经网络&#xff1a;超参数调节&#xff0c;正则化&#xff0c;优化 之前已经学习了如何构建神经网络&#xff0c;本章将继续学习如何有效运行神经网络&#xff0c;内容涉及超参数调优&#xff0c;如何构建数据以及如何确保优化算法快速运行&#xff0c;从而使学习算…

LVM卷在线扩容报错:resize would cause inodes_count overflow

一、问题描述 某次在线环境&#xff0c;存储使用率告警在线扩容时&#xff0c;文件系统扩容失败&#xff0c;报错如下&#xff1a; Size of logical volume sihua/video changed from <162.00 TiB (42467321 extents) to <258.00 TiB (67633142 extents).Logical volu…

[附源码]计算机毕业设计付费自习室管理小程序Springboot程序

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; Springboot mybatis MavenVue等等组成&#xff0c;B/S模式…

【JDBC上篇】什么是JDBC

文章目录1、对JDBC本质的理解2、代码模拟JDBC的本质3、JDBC编程的六步4、通过类加载的方式注册驱动5、属性配置文件与JDBC6、处理查询结果集1、对JDBC本质的理解 Java DataBase Connectivity&#xff08;Java语言连接数据库&#xff09;&#xff0c;其本质是SUN公司指定的一套…

CSP22.3 T4通信系统管理

之前在CCF CSP认证2022年3月完整题解这篇博客记录了自己花了两天时间乱搞出来的方法&#xff0c;但是实际上动态维护区间最值&#xff0c;通过setsetset实现会更简洁&#xff0c;用优先队列需要额外开数组记录堆中节点的有效性。 而且在处理额度失效上&#xff0c;我也使用了最…

教你用响应式建站平台搭建网站

响应式网站搭建大家知道是什么吗&#xff1f;我们可以经常听到PC端网站、移动端网站&#xff0c;这些就是为特定终端而制作的网站版本&#xff0c;而响应式网站就是一个网站能够兼容多个终端&#xff0c;而不是为每个终端做一个特定的版本。那么我们怎么用响应式建站平台搭建网…

基于极限学习机(ELM)进行多变量用电量预测(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️❤️&#x1f4a5;&#x1f4a5;&#x1f4a5; &#x1f389;作者研究&#xff1a;&#x1f3c5;&#x1f3c5;&#x1f3c5;本科计算机专业&#xff0c;研究生电气学硕…

使用adb命令导出hprof文件

我们可以使用android studio profiler导入hprof文件分析android应用的内存问题。那么如何从手机上面导出这份文件呢&#xff1f; 把应用的heap 区的数据取出来保存在一个hprof文件上&#xff0c;然后把hprof文件导入到可以查看这个文件的工具上&#xff08;如android studio p…

Spring Security总结

目录 介绍 项目搭建 Security认证 UserDetailsService 内存认证 数据库认证 PasswordEncoder密码解析器 自定义登录页面 退出登录 CSRF防护 Remember me Security授权 RBAC 权限表设计 查询访问权限 配置类设置访问权限 自定义访问控制逻辑 注解设置访问权限 Secured Pre…

29.前端笔记-HTML-Html5的新特性

目录1、HtML5新增的语义化标签2、HTML5新增的多媒体标签&#xff08;1&#xff09;音频标签&#xff1a;< audio>audio的常见属性&#xff08;2&#xff09;视频标签&#xff1a;< video>video常见属性3、新增表单元素input的类型type4、新增表单属性1、HtML5新增的…

R语言中的生存分析Survival analysis晚期肺癌患者4例

第1部分&#xff1a;生存分析简介 最近我们被客户要求撰写关于生存分析的研究报告&#xff0c;包括一些图形和统计输出。本演示文稿将介绍生存分析 &#xff0c;参考&#xff1a; Clark, T., Bradburn, M., Love, S., & Altman, D. (2003). Survival analysis part I: Ba…

06 数学软件与建模---拟合

一、知识储备 1.曲线拟合问题的提法 已知一组&#xff08;二维&#xff09;数据&#xff0c;即平面上 n个点&#xff08;xi,yi) i1,…,n, 寻求一个函数&#xff08;曲线&#xff09;yf(x), 使 f(x) 在某种准则下与所有数据点最为接近&#xff0c;即曲线拟合得最好&#xff0…