使用Java语言中多态、继承、封装、接口等知识点综合来设计与实现图书管理系统。
利用封装将书籍的属性:书名、作者、价格、是否被借出设为private权限;提供对应的 get 方法、set 方法、 空参构造器、和一个boolean类型的是否被借出的构造器(在这里可以不进行初始化,默认为false),再提供一个toString() 方法的重写来输出书籍这个数组。
代码展示:
package book;
public class Book {
private String name;//书名
private String author;//作者
private int price;//价格
private String type;//书的类型
private boolean isBorrowed;//是否被借出 默认值是false
public Book(String name, String author, int price, String type) {
this.name = name;
this.author = author;
this.price = price;
this.type = type;
}
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) ? " ,已借阅" : " , 未借出")+
//", isBorrowed=" + isBorrowed +
'}';
}
}
在BookList类提供一个Book类型的books数组来存储书籍,同时在空参构造器中进行初始化添加3本书,再提供对应的 get 方法和 set 方法。
代码展示:
package book;
public class BookList {
private Book[] books;
private int usedSize; //记录当前书架放了几本书
public BookList() {
this.books = new Book[10];
this.books[0] = new Book("三国演义","罗贯中",10,"小说");
this.books[1] = new Book("西游记","吴承恩",9,"小说");
this.books[2] = new Book("红楼梦","曹雪芹",8,"小说");
this.usedSize = 3;
}
public int getUsedSize() {
// 获取数组大小
return usedSize;
}
public void setUsedSize(int usedSize) {
// 设置数组大小
this.usedSize = usedSize;
}
public Book getBook(int pos) {
// 获取数组某个位置上的书籍
return books[pos];
}
public void setbook(Book book,int pos) {
//设置数组某个位置的书籍
books[pos] = book;
}
}
在用户类中定义用户的名字,定义一个抽象方法menu() 来给子类管理员类和普通用户类重写,再设计一个接口类型的数组iOperations,存放后面的图书操作功能。
代码展示:
package user;
import book.BookList;
import operation.IOperation;
public abstract class User {
protected String name;
// 数组没有初始化,因为没有给他分配内存
protected IOperation[] iOperations;
public User(String name) {
this.name = name;
}
public abstract int menu();
public void doOperation(int choice, BookList bookList) {
//this.iOperations[choice].work(bookList);
IOperation iOperation = this.iOperations[choice];
iOperation.work(bookList);
}
}
继承User类,用构造方法 一定会对IOperation数组进行初始化,实现menu() 方法。
代码展示:
package user;
import operation.*;
import java.util.Scanner;
public class AdminUser extends User{
// 会调用构造方法 一定会对IOperation数组进行初始化
public AdminUser(String name) {
super(name);
this.iOperations = new IOperation[] {
new ExitOperation(),
new FindOperation(),
new AddOperation(),
new DelOperation(),
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;
}
}
继承User类,用构造方法 一定会对IOperation数组进行初始化,实现menu() 方法。
代码展示:
package user;
import operation.*;
import java.util.Scanner;
public class NormalUser extends User{
public NormalUser(String name) {
super(name);
this.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;
}
}
实现图书查找、新增等操作都需要遵循一定的规范,而这个规范由接口来实现,设计一个work() 方法,并提供对应的work() 方法供接口的实现类,即对各种操作进行重写。
代码展示:
package operation;
import book.Book;
import book.BookList;
public interface IOperation {
void work(BookList bookList);
}
引用接口,重写work() 方法,用到 .equals()方法。
思路:
代码展示:
package operation;
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();
int currentSize = bookList.getUsedSize();
//Book book = bookList[i];
// bookList不是数组 是类 有一个Book数组是private拿不到
// 提供一个方法拿到books数组的i下标
for (int i = 0; i < currentSize; i++) {
Book book = bookList.getBook(i);
if (book.getName().equals(name)) {
System.out.println("存在这本书,信息如下:");
System.out.println(book);
return;
}
}
System.out.println("没有你要找的这本书:"+name);
}
}
将操作类实现操作接口,同时为重写的方法提供书架类,便于操作。
思路:
package operation;
import book.Book;
import book.BookList;
import java.util.Scanner;
public class AddOperation implements IOperation{
public void work(BookList bookList) {
System.out.println("新增图书");
System.out.println("请输入你要新增的图书的书名:");
Scanner scanner = new Scanner(System.in);
String name = scanner.nextLine();
System.out.println("请输入你要新增的图书的作者:");
String author = scanner.nextLine();
System.out.println("请输入你要新增的图书的价格:");
int price = scanner.nextInt();
System.out.println("请输入你要新增的图书的类型:");
//scanner.nextLine(); // 多读一次
//String type = scanner.nextLine(); 会把输入价格后的回车给吞进去 就不能输入type
String type = scanner.next();
Book book = new Book(name,author,price,type);
int currentSize = bookList.getUsedSize();
for (int i = 0; i < currentSize; i++) {
Book tmp = bookList.getBook(i);
if (tmp.getName().equals(name)) {
System.out.println("已经存放了这本书,不能重复添加");
return;
}
}
//没有重复的书 开始新增
bookList.setbook(book,currentSize);//调用setbook方法存放书籍和位置
bookList.setUsedSize(currentSize+1);//书架大小+1
}
}
与增加同理,引用IOperation接口,重写work()方法。删除就是移动后面的数据进行覆盖,最后size–。
思路:
package operation;
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();
int i = 0;
int index = -1;
for (; i < currentSize; i++) {
Book tmp = bookList.getBook(i);
if (tmp.getName().equals(name)) {
index = i;
break; //记录下来了要删除图书的下标
}
}
if (i >= currentSize) {
System.out.println("没找到你要删除的书名:"+name);
return;
}
// 可以删除了 移动后面的数据进行覆盖 最后size--
for (int j = index; j < currentSize-1; j++) {
//bookList[j] = bookList[j+1] 报错 bookList不是数组是一个类
//调用bookList类里getBook拿到后面一本书j+1,再调用setBook
// 把后一本书放进前一本书里面j
Book book = bookList.getBook(j+1);
bookList.setbook(book,j);
}
//因为删除一本书把后面的数据往前覆盖,最后一个内存要回收
bookList.setbook(null,currentSize-1);
bookList.setUsedSize(currentSize-1);
System.out.println("删除成功");
}
}
引用IOperation接口,重写work()方法。
思路:
代码展示:
package operation;
import book.Book;
import book.BookList;
public class ShowOperation implements IOperation{
@Override
public void work(BookList bookList) {
System.out.println("显示图书");
int currentSize = bookList.getUsedSize();
for (int i = 0; i < currentSize; i++) {
Book book = bookList.getBook(i);
System.out.println(book);
}
}
}
引用接口,重写work方法。
在Java中,System.exit(0);输入0即可退出。
package operation;
import book.BookList;
public class ExitOperation implements IOperation{
@Override
public void work(BookList bookList) {
System.out.println("退出系统");
//应该要对bookList 资源回收
System.exit(0);
}
}
引用接口,重写work方法。
思路:
代码展示:
package operation;
import book.Book;
import book.BookList;
import java.util.Scanner;
public class BorrowOperation 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);
if (book.getName().equals(name)) {
System.out.println("借阅成功");
book.setBorrowed(true);
return;
}
}
System.out.println("没有你要借阅的图书"+name);
}
}
引用接口,重写work方法。
思路:
代码展示:
package operation;
import book.Book;
import book.BookList;
import com.sun.xml.internal.ws.policy.privateutil.PolicyUtils;
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);
if (book.getName().equals(name)) {
System.out.println("归还成功");
book.setBorrowed(false);
return;
}
}
System.out.println("没有你要归还的图书"+name);
}
}
以上所有的框架都搭建好了,点击main方法测试:
import book.BookList;
import operation.IOperation;
import user.AdminUser;
import user.NormalUser;
import user.User;
import java.util.Scanner;
public class Main {
public static User login() {
System.out.println("请输入你的名字:");
Scanner scanner = new Scanner(System.in);
String name = scanner.nextLine();
System.out.println("请输入你的身份,1:管理员 2:普通用户-》");
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();
//发生多态绑定 引用的对象不一样 调用menu()所表现的行为不一样
// user = new AdminUser(name)
// user = new NormalUser(name);
User user = login();
while (true) {
// 父类引用子类对象,调用menu(),要父类有menu(),子类重写方法
int choice = user.menu();
//根据你菜单返回的choice来执行对应的操作
user.doOperation(choice,bookList);
}
}
}
源码链接: 源码,点击跳转https://gitee.com/xiao-jiahao6/java/tree/master/TestBook/src