开闭原则
开闭原则定义
开闭原则是Java世界里最基础的设计原则,它指导我们如何建立一个稳定的、灵活的系统,定义如下:
Software entities like classes,modules and functions should be open for extension but closed for modifications.(一个软件实体如类、模块和函数应该对扩展开放,对修改关闭)
开闭原则应用案例说明
我们以书店销售书籍为例子,先画下类图:
IBook定义了数据的3个属性:名称、价格和作者。小说类NovelBook是一个具体的实现类,是所有小说书籍的总称,BookStore指的是书店,代码如下:
package com.yuqingit.pattern1;
/**
* @version V1.0
* @description:
* @author: yuqing
* @date: 2020/9/17 8:18
*/
public interface IBook {
/**
* 名称
* @return
*/
String getName();
/**
* 售价
* @return
*/
int getPrice();
/**
* 作者
* @return
*/
String getAuthor();
}
#NovelBook 代码如下:
package com.yuqingit.pattern1;
/**
* @version V1.0
* @description:
* @author: yuqing
* @date: 2020/9/17 8:22
*/
public class NovelBook implements IBook{
/** 名称 */
private String name;
/** 价格 */
private int price;
/** 作者 */
private String author;
public NovelBook(String name, int price, String author) {
this.name = name;
this.price = price;
this.author = author;
}
@Override
public String getName() {
return this.name;
}
@Override
public int getPrice() {
return this.price;
}
@Override
public String getAuthor() {
return this.author;
}
}
#BookStore 代码如下:
package com.yuqingit.pattern1;
import java.util.ArrayList;
/**
* @version V1.0
* @description:
* @author: yuqing
* @date: 2020/9/17 8:27
*/
public class BookStore {
private static ArrayList<IBook> bookList = new ArrayList<>();
static {
bookList.add(new NovelBook("天龙八部",3200,"金庸"));
bookList.add(new NovelBook("巴黎圣母院",5600,"雨果"));
bookList.add(new NovelBook("悲惨世界",3500,"雨果"));
}
public static void main(String[] args) {
System.out.println("----------------书店卖出去的书籍如下:---------------------");
for (IBook book : bookList){
System.out.println("书籍名称:"+ book.getName() +" 书籍作者:" + book.getAuthor() + " 书籍价格:" + book.getPrice() + "分");
}
}
}
假如这时候书店要进行促销活动,针对小说类的书籍进行打折,如所有小说类书籍8折,那这样价格就会变动,此时我就需要调整代码,很多人会说这还不简单,直接在实现类获取价格方法类修改,立刻满足需求,修改后代码如下:
package com.yuqingit.pattern1;
/**
* @version V1.0
* @description:
* @author: yuqing
* @date: 2020/9/17 8:22
*/
public class NovelBook implements IBook{
/** 名称 */
private String name;
/** 价格 */
private int price;
/** 作者 */
private String author;
public NovelBook(String name, int price, String author) {
this.name = name;
this.price = price;
this.author = author;
}
@Override
public String getName() {
return this.name;
}
@Override
public int getPrice() {
return this.price*80/100;
}
@Override
public String getAuthor() {
return this.author;
}
}
我们仔细想想,是不是有更好的方式去满足此需求,如果这样调整了,如果要看书本原有的价格呢?显然看不了。我们就想是不是通过扩展NovelBook 来满足此需求,也即通过继承此类,覆盖getPrice方法来满足,这样是不是就不影响原有的业务(看书的原价格)。调整后的类图如下:
OK,至此我们就阐述了下开闭原则的实现,对扩展开放,对修改关闭。