文章摘要(AI生成)
外观/门面模式提供了一个统一的高层次接口,使得子系统更易于使用,减少了系统的相互依赖,提高了灵活性和安全性。然而,其缺点是不符合开闭原则,对系统修改比较困难。使用场景包括复杂模块或子系统的外界访问、预防低水平人员带来的风险等。最佳实践是将复杂系统封装为一个门面,提高项目结构简单性和扩展性。在示例中,通过多个角色完成从邮件中获取用户名字的数据库类、编写html文件的类以及提供高层接口的窗口类,展示了门面模式在实际应用中的作用。通过门面模式的使用,可以约束项目成员的代码质量,提升整体项目质量。
外观/门面模式
定义
要求一个子系统的外部与其内部的通信必须通 过一个统一的对象进行。门面模式提供一个高层次的接口,使得子系统更易于使用。
优缺点
优点
-
减少系统的相互依赖
想想看,如果我们不使用门面模式,外界访问直接深入到子系统内部,相互之间是一种强耦合关系,你死我就死,你活我才能活,这样的强依赖是系统设计所不能接受的,门面模式的出现就很好地解决了该问题,所有的依赖都是对门面对象的依赖,与子系统无关。
-
提高了灵活性
依赖减少了,灵活性自然提高了。不管子系统内部如何变化,只要不影响到门面对象,任你自由活动。
-
提高安全性
想让你访问子系统的哪些业务就开通哪些逻辑,不在门面上开通的方法,你休想访问到。
缺点
门面模式最大的缺点就是不符合开闭原则,对修改关闭,对扩展开放,看看我们那个门 面对象吧,它可是重中之重,一旦在系统投产后发现有一个小错误,你怎么解决?完全遵从 开闭原则,根本没办法解决。继承?覆写?都顶不上用,唯一能做的一件事就是修改门面角 色的代码,这个风险相当大,这就需要大家在设计的时候慎之又慎,多思考几遍才会有好收获
使用场景
-
为一个复杂的模块或子系统提供一个供外界访问的接口
-
子系统相对独立——外界对子系统的访问只要黑箱操作即可
比如利息的计算问题,没有深厚的业务知识和扎实的技术水平是不可能开发出该子系统的,但是对于使用该系统的开发人员来说,他需要做的就是输入金额以及存期,其他的都不用关心,返回的结果就是利息,这时候,门面模式是非使用不可了。
-
预防低水平人员带来的风险扩散
比如一个低水平的技术人员参与项目开发,为降低个人代码质量对整体项目的影响风险,一般的做法是"画地为牢”,只能在指定的子系统中开发,然后再提供门面接口进行访问操作。
最佳实践
门面模式是一个很好的封装方法,一个子系统比较复杂时,比如算法或者业务比较复杂,就可以封装出一个或多个门面出来,项目的结构简单,而且扩展性非常好。还有,对于 一个较大项目,为了避免人员带来的风险,也可以使用门面模式,技术水平比较差的成员, 尽量安排独立的模块,然后把他写的程序封装到一个门面里,尽量让其他项目成员不用看到 这些人的代码,看也看不懂,我也遇到过一个“高人”写的代码,private方法、构造函数、常 量基本都不用,你要一个public方法,好,一个类里就一个public方法,所有代码都在里面, 然后你就看吧,一大坨程序,看着就能把人逼疯。使用门面模式后,对门面进行单元测试, 约束项目成员的代码质量,对项目整体质量的提升也是一个比较好的帮助。
角色
-
窗口类(facade):代表构成系统的许多其他角色的简单窗口。向系统外部提供统一API
-
构成系统的许多其他角色:这些角色各自完成自己的工作,并不知道facade角色
-
请求者(client):负责调用facade角色
spring实现
在以下示例中,我们将要编写一个web页面,编写过程需要有从邮件中获取用户名字的数据库类,编写html文件的类,以及一个提供高层接口的类
多角色
从邮件中获取用户名字的数据库类:
public class Database {
public static Properties getProperties(String dbname){
String filename = dbname + ".txt";
Properties prop = new Properties();
try{
File file = ResourceUtils.getFile("classpath:templates/"+ filename);
prop.load(Files.newInputStream(Paths.get(file.getPath())));
}catch(IOException e){
System.out.println("Warning: "+filename+" is not found");
}
return prop;
}
}
编写html文件的类
@Component
public class HtmlWriter {
private Writer writer ;
public void setWriter(Writer writer) {
this.writer = writer;
}
public void title(String title) throws IOException{
writer.write("<html>");
writer.write("<head>");
writer.write("<title>"+title+"</title>");
writer.write("</head>");
writer.write("<body>\n");
writer.write("<h1>"+title+"</h1>\n");
}
public void paragraph(String msg) throws IOException{
writer.write("<p>"+msg+"</p>\n");
}
public void link(String href,String caption) throws IOException{
paragraph("<a href=\""+href+"\">"+caption+"</a>");
}
public void mailto(String mailaddr,String username) throws IOException{
link("mailto:"+mailaddr, username);
}
public void close() throws IOException{
writer.write("</body>");
writer.write("</html>\n");
writer.close();
}
}
窗口类
提供统一的创建邮件欢迎页面的接口
@Component
@Slf4j
public class PageMaker {
@Resource
private HtmlWriter writer;
public void makeWelcomePage(String mailaddr,String filename, String target){
try{
Properties mailprop = Database.getProperties(filename);
String username = mailprop.getProperty(mailaddr);
File file = ResourceUtils.getFile("classpath:templates/"+ target + ".html");
writer.setWriter(new FileWriter(file));
writer.title("Welcome to "+username+"'s page");
writer.paragraph(username+"欢迎来到"+username+"的主页");
writer.paragraph("等着你的邮件");
writer.mailto(mailaddr, username);
writer.close();
log.info(target+" is created for "+mailaddr+" ("+username+")");
}catch(Exception e){
e.printStackTrace();
}
}
}
测试
@Controller
public class FacadeController {
@Resource
private PageMaker pageMaker;
@GetMapping("/facade")
public String facade(){
pageMaker.makeWelcomePage("hyuki@hyuki.com", "maildata", "hello");
return "hello";
}
}
测试结果为
http://localhost:8080/facade
HTTP/1.1 200
Content-Type: text/html;charset=UTF-8
Content-Language: zh-CN
Transfer-Encoding: chunked
Date: Sun, 21 Aug 2022 08:50:24 GMT
Keep-Alive: timeout=60
Connection: keep-alive
<html>
<head><title>Welcome to Hiroshi Yuki's page</title></head>
<body>
<h1>Welcome to Hiroshi Yuki's page</h1>
<p>Hiroshi Yuki欢迎来到Hiroshi Yuki的主页</p>
<p>等着你的邮件</p>
<p><a href="mailto:hyuki@hyuki.com">Hiroshi Yuki</a></p>
</body>
</html>
享元模式
定义
运用共享对象有效地支持大量细粒度的对象
优点
享元模式是一个非常简单的模式,它可以大大减少应用程序创建的对象,降低程序内存 的占用,增强程序的性能,但它同时也提高了系统复杂性,需要分离出外部状态和内部状态,而且外部状态具有固化特性,不应该随内部状态改变而改变,否则导致系统的逻辑混乱。
使用场景
在如下场景中则可以选择使用享元模式。
- 系统中存在大量的相似对象。
- 细粒度的对象都具备较接近的外部状态,而且内部状态与环境无关,也就是说对象没有特定身份。
- 需要缓冲池的场景。
最佳实践
享元模式在Java API中也是随处可见,String类的intern方法就是一个很好的例子
虽然可以使用享元模式可以实现对象池,但是这两者还是有比较大 的差异,对象池着重在对象的复用上,池中的每个对象是可替换的,从同一个池中获得A对 象和B对象对客户端来说是完全相同的,它主要解决复用,而享元模式在主要解决的对象的 共享问题,如何建立多个可共享的细粒度对象则是其关注的重点
角色
-
享元角色(flyweight):指的是会被共享的实例
-
享元工厂(flyweightFactory):生成享元实例的工厂
-
请求者(client):请求调用工厂生成享元角色
spring实现
将许多普通字符组合成为大型字符,根据我们的输入以供程序打印使用
享元角色
表示大型字符的类
@Component
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
@Slf4j
public class BigChar {
private char charname;
private String fontdata;
public void initBigChar(char charname){
this.charname = charname;
try{
File file = ResourceUtils.getFile("classpath:static/big"+charname+".txt");
BufferedReader reader = new BufferedReader(new FileReader(file));
String line;
StringBuffer buf = new StringBuffer();
while((line=reader.readLine())!=null){
buf.append(line);
buf.append("\n");
}
reader.close();
this.fontdata = buf.toString();
}catch(IOException e){
this.fontdata = charname+"?";
}
}
public void print(){
log.info(fontdata);
}
}
享元工厂
存储并生成大型字符的工厂
@Component
public abstract class BigCharFactory {
private HashMap<String, BigChar> pool = new HashMap<String, BigChar>();
@Lookup
public abstract BigChar bigChar();
public BigChar genBigChar(char charName){
BigChar bigChar = bigChar();
bigChar.initBigChar(charName);
return bigChar;
}
/**
* 类似与spring中beanfactory的创建
* @param charname
* @return
*/
public synchronized BigChar getBigChar(char charname){
BigChar bc = pool.get(""+charname);
if(bc==null){
bc = genBigChar(charname);
pool.put(""+charname, bc);
}
return bc;
}
}
使用者
表示多个bigchar组成的大型字符串的类
@Component
public class BigString {
private BigChar[] bigchars;
@Resource
private BigCharFactory factory;
public void initBigString(String string) {
bigchars = new BigChar[string.length()];
for(int i =0 ;i<bigchars.length;i++){
bigchars[i] = factory.getBigChar(string.charAt(i));
}
}
public void print(){
for(int i=0;i<bigchars.length;i++){
bigchars[i].print();
}
}
}
测试
@RestController
public class FlyweightController {
@Resource
private BigString bigString;
@GetMapping("/flyweight")
public String flyweight(String param){
bigString.initBigString(param);
bigString.print();
return "success";
}
}
当请求为GET http://localhost:8080/flyweight?param=12113
时,输出结果为:
2022-08-21 17:13:58.179 INFO 23216 --- [nio-8080-exec-1] c.e.designpattern.Flyweight.BigChar : ......##........
..######........
......##........
......##........
......##........
......##........
..##########....
................
2022-08-21 17:13:58.179 INFO 23216 --- [nio-8080-exec-1] c.e.designpattern.Flyweight.BigChar : ....######......
..##......##....
..........##....
......####......
....##..........
..##............
..##########....
................
2022-08-21 17:13:58.179 INFO 23216 --- [nio-8080-exec-1] c.e.designpattern.Flyweight.BigChar : ......##........
..######........
......##........
......##........
......##........
......##........
..##########....
................
2022-08-21 17:13:58.179 INFO 23216 --- [nio-8080-exec-1] c.e.designpattern.Flyweight.BigChar : ......##........
..######........
......##........
......##........
......##........
......##........
..##########....
................
2022-08-21 17:13:58.179 INFO 23216 --- [nio-8080-exec-1] c.e.designpattern.Flyweight.BigChar : ....######......
..##......##....
..........##....
......####......
..........##....
..##......##....
....######......
................
代理模式
定义
为其他对象提供一种代理以控制对这个对象的访问。
优点
-
职责清晰
真实的角色就是实现实际的业务逻辑,不用关心其他非本职责的事务,通过后期的代理完成一件事务,附带的结果就是编程简洁清晰。
-
高扩展性
具体主题角色是随时都会发生变化的,只要它实现了接口,甭管它如何变化,都逃不脱如来佛的手掌(接口),那我们的代理类完全就可以在不做任何修改的情况下使用。
-
智能化
例如Struts通过动态代理把表单元素映射到对象
使用场景
代理模 式的使用场景非常多,大家可以看看Spring AOP,这是一个非常典型的动态代理
最佳实践
代理模式应用得非常广泛,大到一个系统框架、企业平台,小到代码片段、事务处理, 稍不留意就用到代理模式。可能该模式是大家接触最多的模式,而且有了AOP大家写代理就 更加简单了,有类似Spring AOP和AspectJ这样非常优秀的工具,拿来主义即可!不过,大家可以看看源代码,特别是调试时,只要看到类似$Proxy0这样的结构,你就应该知道这是一 个动态代理了。
角色
- 主体(subject):定义了是代理角色和实际角色之间具有一致性的接口
- 代理人(proxy):尽量处理请求,只在必要时生成实际角色
- 实际主体(realSubject):在代理人无法胜任时才会用到
- 请求者:使用代理模式的角色
静态代理实现
下面我们来看一段使用了Proxy模式的示例程序。这段示例程序实现了一个“带名字的打印机”。说是打印机,其实只是将文字显示在界面上而已。在Main类中会生成PrinterProxy类的实例(即“代理人”)。首先我们会给实例赋予名字Alice并在界面中显示该名字。接着会将实例名字改为Bob,然后显示该名字。在设置和获取名字时,都不会生成真正的Printer类的实例(即本人),而是由PrinterProxy类代理。最后,直到我们调用print方法,开始进入实际打印阶段后,PrinterProxy类才会生成Printer类的实例。
主体
public interface Printable {
void setPrinterName(String name);
String getPrinterName();
void print(String string);
}
实际主体
@Slf4j
public class Printer implements Printable {
private String name;
public Printer(String name) {
super();
this.name = name;
heavyJob("Printer的实例生成中("+name+")");
}
private void heavyJob(String msg) {
// TODO Auto-generated method stub
System.out.println(msg);
for(int i =0;i<5;i++){
try{
Thread.sleep(1000);
}catch(InterruptedException e){
}
log.info(".");
}
log.info("结束。");
}
@Override
public void setPrinterName(String name) {
this.name = name;
}
@Override
public String getPrinterName() {
// TODO Auto-generated method stub
return name;
}
@Override
public void print(String string) {
log.info("=== "+name+" ===");
log.info(string);
}
}
代理人
public class PrinterProxy implements Printable {
private String name;
private Printer real;
public PrinterProxy(String name) {
super();
this.name = name;
}
@Override
public void setPrinterName(String name) {
if(real!=null){
real.setPrinterName(name);
}
this.name = name;
}
@Override
public String getPrinterName() {
// TODO Auto-generated method stub
return name;
}
@Override
public void print(String string) {
realize();
real.print(string);
}
private synchronized void realize() {
// TODO Auto-generated method stub
if(real == null){
real = new Printer(name);
}
}
}
调用者
public class Main {
public static void main(String[] args) {
Printable p =new PrinterProxy("Alice");
System.out.println("现在的名字是"+p.getPrinterName()+"。");
p.setPrinterName("Bob");
System.out.println("现在的名字是"+p.getPrinterName()+"。");
p.print("hello, world.");
}
}
装饰器模式
定义
动态地给一个对象添加一些额外的职责或者行为
优缺点
优点
- 装饰类和被装饰类可以独立发展,而不会相互耦合。换句话说,Component类无须知 道Decorator类,Decorator类是从外部来扩展Component类的功能,而Decorator也不用知道具 体的构件。
- 装饰模式是继承关系的一个替代方案。我们看装饰类Decorator,不管装饰多少层,返 回的对象还是Component,实现的还是is-a的关系。
- 装饰模式可以动态地扩展一个实现类的功能,这不需要多说,装饰模式的定义就是如 此。
缺点
对于装饰模式记住一点就足够了:多层的装饰是比较复杂的。为什么会复杂呢?你想想 看,就像剥洋葱一样,你剥到了最后才发现是最里层的装饰出现了问题,想象一下工作量 吧,因此,尽量减少装饰类的数量,以便降低系统的复杂度。
使用场景
- 需要扩展一个类的功能,或给一个类增加附加功能。
- 需要动态地给一个对象增加功能,这些功能可以再动态地撤销。
- 需要为一批的兄弟类进行改装或加装功能,当然是首选装饰模式
最佳实践
装饰模式是对继承的有力补充。你要知道继承不是万能的,继承可以解决实际的问题, 但是在项目中你要考虑诸如易维护、易扩展、易复用等,而且在一些情况下你要是用继承就会增加很多子类,而且灵活性非常差,那当然维护也不容易了,也就是说装饰模式可以替代继承,解决我们类膨胀的问题。同时,你还要知道继承是静 态地给类增加功能,而装饰模式则是动态地增加功能
装饰模式还有一个非常好的优点:扩展性非常好。在一个项目中,你会有非常多的因素 考虑不到,特别是业务的变更,不时地冒出一个需求,尤其是提出一个令项目大量延迟的需 求时,那种心情是相当的难受!装饰模式可以给我们很好的帮助,通过装饰模式重新封装一 个类,而不是通过继承来完成
角色
- 组件实例(Component):增加功能时的核心角色。
- 具体组件(ConcreteComponent):该角色是实现了Component角色所定义的接口(API)的具体实例。
- 装饰器(Decorator):该角色具有与Component 角色相同的接口(API)。在它内部保存了被装饰对象——Component角色。Decorator角色知道自己要装饰的对象。
- 具体装饰器(ConcreteDecorator):该角色是具体的Decorator角色。
spring实现
本示例程序的功能是给文字添加装饰边框。
组件实例
用于显示字符串的抽象类
@Slf4j
public abstract class Display {
public abstract int getColumns();
public abstract int getRows();
public abstract String getRowText(int row);
public final void show(){
for(int i=0;i<getRows();i++){
log.info(getRowText(i));
}
}
}
具体组件
用于显示单行字符串的类
@Component
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class StringDisplay extends Display {
private String string;
public void initStringDisplay(String string) {
this.string = string;
}
@Override
public int getColumns() {
// TODO Auto-generated method stub
return string.getBytes().length;
}
@Override
public int getRows() {
// TODO Auto-generated method stub
return 1;
}
@Override
public String getRowText(int row) {
// TODO Auto-generated method stub
if (row == 0) {
return string;
} else {
return null;
}
}
}
装饰器
用于显示边框的抽象类
public abstract class Border extends Display {
protected Display display;
protected void initBorder(Display display) {
this.display = display;
}
}
具体装饰器
用于显示上下左右边框的类
@Component
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class FullBorder extends Border{
protected void initFullBorder(Display display) {
initBorder(display);
// TODO Auto-generated constructor stub
}
@Override
public int getColumns() {
// TODO Auto-generated method stub
return 1+display.getColumns()+1;
}
@Override
public int getRows() {
// TODO Auto-generated method stub
return 1+display.getRows()+1;
}
@Override
public String getRowText(int row) {
// TODO Auto-generated method stub
if(row==0){
return "+"+makeLine('-',display.getColumns())+"+";
}else if(row == display.getRows()+1){
return "+"+makeLine('-', display.getColumns())+"+";
}else{
return "|"+display.getRowText(row-1)+"|";
}
}
private String makeLine(char ch,int count){
StringBuffer buf = new StringBuffer();
for (int i = 0; i < count; i++) {
buf.append(ch);
}
return buf.toString();
}
}
只显示左右边框的类
@Component
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class SideBorder extends Border{
private char borderChar;
public void initSideBorder(Display display,char ch) {
initBorder(display);
// TODO Auto-generated constructor stub
this.borderChar = ch;
}
@Override
public int getColumns() {
// TODO Auto-generated method stub
return 1+display.getColumns()+1;
}
@Override
public int getRows() {
// TODO Auto-generated method stub
return display.getRows();
}
@Override
public String getRowText(int row) {
// TODO Auto-generated method stub
return borderChar+display.getRowText(row)+borderChar;
}
}
实例生成器
由具体组件或装饰器生成组件的生成器
@Component
public abstract class DisplayGenerator {
@Lookup
protected abstract StringDisplay stringDisplay();
@Lookup
protected abstract SideBorder sideBorder();
@Lookup
protected abstract FullBorder fullBorder();
public Display getStringDisplay(String s){
StringDisplay stringDisplay = stringDisplay();
stringDisplay.initStringDisplay(s);
return stringDisplay;
}
public Display getFullBorder(Display display){
FullBorder fullBorder = fullBorder();
fullBorder.initFullBorder(display);
return fullBorder;
}
public Display getSideBorder(Display display, char c){
SideBorder sideBorder = sideBorder();
sideBorder.initSideBorder(display, c);
return sideBorder;
}
}
测试
@RestController
public class DecoratorController {
@Resource
DisplayGenerator displayGenerator;
@GetMapping("decorate")
public String decorate(){
Display b1 = displayGenerator.getStringDisplay("hello, world.");
Display b2 = displayGenerator.getSideBorder(b1, '#');
Display b3 = displayGenerator.getFullBorder(b2);
b1.show();
b2.show();
b3.show();
Display b4 = displayGenerator.getSideBorder(displayGenerator.getFullBorder(
displayGenerator.getFullBorder(displayGenerator.getSideBorder(displayGenerator.getFullBorder(displayGenerator.getStringDisplay(
"你好,世界")), '*'))), '/');
b4.show();
return "success";
}
}
测试结果如下:
2022-08-21 18:02:59.615 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : hello, world.
2022-08-21 18:02:59.615 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : #hello, world.#
2022-08-21 18:02:59.615 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : +---------------+
2022-08-21 18:02:59.615 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : |#hello, world.#|
2022-08-21 18:02:59.615 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : +---------------+
2022-08-21 18:02:59.615 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : /+---------------------+/
2022-08-21 18:02:59.616 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : /|+-------------------+|/
2022-08-21 18:02:59.616 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : /||*+---------------+*||/
2022-08-21 18:02:59.616 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : /||*|你好,世界|*||/
2022-08-21 18:02:59.616 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : /||*+---------------+*||/
2022-08-21 18:02:59.616 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : /|+-------------------+|/
2022-08-21 18:02:59.616 INFO 4812 --- [nio-8080-exec-1] c.e.designpattern.Decorator.Display : /+---------------------+/
评论区