设计模式(结构型)
[toc]
结构型设计模式关注如何将现有的类或对象组织在一起形成更加强大的结构。 并且根据我们前面学习的合成复用原则,我们该如何尽可能地使用关联关系来代替继承关系是我们本版块需要重点学习的内容。
适配器模式
在生活中,我们经常遇到这样的一个问题:笔记本太轻薄了,以至于没有RJ45网口和USB A口(比如Macbook为了轻薄甚至全是type-c形式的雷电口) 但是现在我们因为工作需要,又得使用这些接口来连接线缆,这时我们想到的第一个解决方案,就是去买一个转接口(扩展坞), 扩展坞可以将type-c口转换为其他类型的接口供我们使用,实际上这就是一种适配模式。
java
public class AdapterMain {
public static void main(String[] args) {
ClassAdapter adapter = new ClassAdapter();
System.out.println("成功得到:" + adapter.supply());
}
}
class TestSupplier {
public String doSupply() {
return "iPhone 14 Pro Max 1TB 紫色";
}
}
//现在的手机供应商,并不是test方法所需要的那种类型
interface Target {
String supply();
}
/**
* 类适配器模式
* */
class ClassAdapter extends TestSupplier implements Target {
//让我们的适配器继承TestSupplier并且实现Target接口
//接着实现supply方法,直接使用TestSupplier提供的实现
@Override
public String supply() {
return super.doSupply();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
不过,这种实现方式需要占用一个继承坑位,如果此时Target不是接口而是抽像类的话,由于Java不支持多继承,那么就无法实现了。同时根据合成复用原则,我们应该更多的通过合成的方式去实现功能,所以我们来看看第二种,也是用的比较多的一种模式,对象适配器
java
/**
* 对象适配器模式
* 现在不再继承TestSupplier,仅实现Target
*/
class ObjectAdapter implements Target {
TestSupplier supplier;
public ObjectAdapter(TestSupplier supplier) {
this.supplier = supplier;
}
@Override
public String supply() {
return supplier.doSupply();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
桥接模式
相信各位都去奶茶店买过奶茶,在购买奶茶的时候,店员首先会问我们,您需要什么类型的奶茶,比如我们此时点了一杯啵啵芋圆奶茶,接着店员会直接问我们需要大杯、中杯还是小杯,最后还会询问我们需要加什么配料,比如椰果、珍珠等,最后才会给我们制作奶茶。
java
//由具体类型的奶茶实现
interface Tea {
//不同的奶茶返回的类型不同
String getType();
}
//分大杯小杯中杯
interface Size {
String getSize();
}
1
2
3
4
5
6
7
8
9
10
11
2
3
4
5
6
7
8
9
10
11
这时,就可以使用我们的桥接模式了,现在我们面临的问题是,维度太多,不可能各种类型各种尺寸的奶茶都去创建一个类,那么我们就还是单独对这些接口进行简单的扩展,单独对不同的维度进行控制,但是如何实现呢?我们不妨将奶茶的类型作为最基本的抽象类,然后对尺寸、配料等属性进行桥接
java
abstract class AbstractTea {
//尺寸作为桥接属性存放在类中
protected Size size;
//在构造时需要知道尺寸属性
protected AbstractTea(Size size) {
this.size = size;
}
//具体类型依然是由子类决定
public abstract String getType();
}
abstract class RefinedAbstractTea extends AbstractTea {
protected RefinedAbstractTea(Size size) {
super(size);
}
//添加尺寸维度获取方式
public String getSize() {
return size.getSize();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
不过这个抽象类提供的方法还不全面,仅仅只有Tea的getType方法,我们还需要添加其他维度的方法,所以继续编写一个子类
java
abstract class RefinedAbstractTea extends AbstractTea {
protected RefinedAbstractTea(Size size) {
super(size);
}
//添加尺寸维度获取方式
public String getSize() {
return size.getSize();
}
}
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
现在我们只需要单独为Size创建子类即可
java
class Large implements Size {
@Override
public String getSize() {
return "大杯";
}
}
1
2
3
4
5
6
7
2
3
4
5
6
7
现在我们如果需要一个大杯的啵啵芋圆奶茶,只需要
java
//创建一个啵啵芋圆奶茶的子类
class KissTea extends RefinedAbstractTea {
//在构造时需要指定具体的大小实现
protected KissTea(Size size) {
super(size);
}
//返回奶茶类型
@Override
public String getType() {
return "啵啵芋圆奶茶";
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
2
3
4
5
6
7
8
9
10
11
12
13
通过桥接模式,使得抽象和实现可以沿着各自的维度来进行变化,不再是固定的绑定关系。
组合模式
组合模式实际上就是将多个组件进行组合,让用户可以对它们进行一致性处理。比如我们的文件夹,一个文件夹中可以有很多个子文件夹或是文件。
它就像是一个树形结构一样,有分支有叶子,而组合模式则是可以对整个树形结构上的所有节点进行递归处理,比如我们现在希望将所有文件夹中的文件的名称前面都添加一个前缀,那么就可以使用组合模式。
java
/**
* 首先创建一个组件抽象,组件可以包含组件,组件有自己的业务方法
*/
abstract class Component {
//添加子组件
public abstract void addComponent(Component component);
//删除子组件
public abstract void removeComponent(Component component);
//获取子组件
public abstract Component getChild(int index);
//执行对应的业务方法,比如修改文件名称
public abstract void test();
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
java
class Directory extends Component { //目录可以包含多个文件或目录
List<Component> child = new ArrayList<>(); //这里我们使用List来存放目录中的子组件
@Override
public void addComponent(Component component) {
child.add(component);
}
@Override
public void removeComponent(Component component) {
child.remove(component);
}
@Override
public Component getChild(int index) {
return child.get(index);
}
@Override
public void test() {
child.forEach(Component::test); //将继续调用所有子组件的test方法执行业务
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
java
class File extends Component { //文件就相当于是树叶,无法再继续添加子组件了
@Override
public void addComponent(Component component) {
throw new UnsupportedOperationException(); //不支持这些操作了
}
@Override
public void removeComponent(Component component) {
throw new UnsupportedOperationException();
}
@Override
public Component getChild(int index) {
throw new UnsupportedOperationException();
}
@Override
public void test() {
System.out.println("文件名称修改成功!" + this); //具体的名称修改操作
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
装饰模式
装饰模式就像其名字一样,为了对现有的类进行装饰。比如一张相片就一张纸,如果直接贴在墙上,总感觉少了点什么,但是我们给其添加一个好看的相框,就会变得非常对味。装饰模式的核心就在于不改变一个对象本身功能的基础上,给对象添加额外的行为,并且它是通过组合的形式完成的,而不是传统的继承关系。
比如我们现在有一个普通的功能类:
java
abstract class Base { //顶层抽象类,定义了一个test方法执行业务
public abstract void test();
}
class BaseImpl extends Base {
@Override
public void test() {
System.out.println("我是业务方法"); //具体的业务方法
}
}
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
不过现在的实现类太单调了,我们来添加一点装饰上去:
java
class Decorator extends Base { //装饰者需要将装饰目标组合到类中
protected Base base;
public Decorator(Base base) {
this.base = base;
}
@Override
public void test() {
base.test(); //这里暂时还是使用目标的原本方法实现
}
}
class DecoratorImpl extends Decorator { //装饰实现
public DecoratorImpl(Base base) {
super(base);
}
@Override
public void test() { //对原本的方法进行装饰,我们可以在前后都去添加额外操作
System.out.println("装饰方法:我是操作前逻辑");
super.test();
System.out.println("装饰方法:我是操作后逻辑");
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
这样,我们就通过装饰模式对类的功能进行了扩展:
java
public class DecorationModeMain {
public static void main(String[] args) {
Base base = new BaseImpl();
//将Base实现装饰一下
Decorator decorator = new DecoratorImpl(base);
//装饰者还可以嵌套
Decorator outer = new DecoratorImpl(decorator);
decorator.test();
outer.test();
}
}
1
2
3
4
5
6
7
8
9
10
11
2
3
4
5
6
7
8
9
10
11
代理模式
代理模式和装饰模式很像,首先请记住,当无法直接访问某个对象或访问某个对象存在困难时,我们就可以通过一个代理对象来间接访问。
实际上代理在我们生活中处处都存在, 比如手机厂商要去销售手机,但是手机厂商本身没有什么渠道可以大规模地进行售卖,很难与这些消费者进行对接, 这时就得交给代理商去进行出售, 比如Apple在中国的直营店很少,但是在中国的授权经销商却很多,手机厂商通过交给旗下代理商的形式来进行更大规模的出售。 比如我们经常要访问Github,但是直接连接会发现很难连的上,这时我们加了一个代理就可以轻松访问,也是在体现代理的作用。
同时,代理类需要保证客户端使用的透明性,也就是说操作起来需要与原本的真实对象相同, 比如我们访问Github只需要输入网址即可访问,而添加代理之后,也是使用同样的方式去访问Github,所以操作起来是一样的。 包括Spring框架其实也是依靠代理模式去实现的AOP记录日志等。
java
abstract class Subject {
public abstract void test();
}
//此类无法直接使用,需要我们进行代理
class SubjectImpl extends Subject {
@Override
public void test() {
System.out.println("我是测试方法!");
}
}
//为了保证和Subject操作方式一样,保证透明性,也得继承
class Proxy extends Subject {
//被代理的对象(甚至可以多重代理)
Subject target;
public Proxy(Subject subject) {
this.target = subject;
}
//由代理去执行被代理对象的方法,并且我们还可以在前后添油加醋
@Override
public void test() {
System.out.println("代理前绕方法");
target.test();
System.out.println("代理后绕方法");
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
对装饰器模式来说,装饰者和被装饰者都实现同一个接口/抽象类。 对代理模式来说,代理类和被代理的类都实现同一个接口/抽象类,在结构上确实没有啥区别。 但是他们的作用不同,装饰器模式强调的是增强自身,在被装饰之后你能够在被增强的类上使用增强后的功能,增强后你还是你,只不过被强化了而已; 代理模式强调要让别人帮你去做事情,以及添加一些本身与你业务没有太多关系的事情(记录日志、设置缓存等)重点在于让别人帮你做。
装饰模式和代理模式的不同之处在于思想。
JDK动态代理
我们还可以使用JDK为我们提供的动态代理机制,我们不再需要手动编写继承关系创建代理类, 它能够在运行时通过反射机制为我们自动生成代理类: JDK 动态代理只能代理接口
java
interface ISubject {
void test();
}
//此类无法直接使用,需要我们进行代理
class SubjectImpl implements ISubject {
@Override
public void test() {
System.out.println("我是测试方法!");
}
}
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
接着我们需要创建一个动态代理的处理逻辑:
java
class TestProxy implements InvocationHandler { //代理类,需要实现InvocationHandler接口
private final Object object; //这里需要保存一下被代理的对象,下面需要用到
public TestProxy(Object object) {
this.object = object;
}
//此方法就是调用代理对象的对应方法时会进入,这里我们就需要编写如何进行代理了
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//method就是调用的代理对象的哪一个方法,args是实参数组
//proxy就是生成的代理对象了,我们看看是什么类型的
System.out.println("代理的对象:" + proxy.getClass());
//在代理中调用被代理对象原本的方法,因为你是代理,还是得执行一下别人的业务,当然也可以不执行,但是这样就失去代理的意义了,注意要用上面的object
Object res = method.invoke(object, args);
//看看返回值是什么
System.out.println("方法调用完成,返回值为:" + res);
//返回返回值
return res;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
Cglib动态代理
由于CGlib底层使用ASM框架进行字节码编辑,所以能够实现不仅仅局限于对接口的代理
xml
<dependency>
<groupId>cglib</groupId>
<artifactId>cglib</artifactId>
<version>3.1</version>
</dependency>
1
2
3
4
5
6
2
3
4
5
6
java
//首先还是编写我们的代理逻辑
public class TestProxy implements MethodInterceptor {
//这些和之前JDK动态代理写法是一样的
private final Object target;
public TestProxy(Object target) {
this.target = target;
}
//我们也是需要在这里去编写我们的代理逻辑
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("现在是由CGLib进行代理操作!" + o.getClass());
//也是直接调用代理对象的方法即可
return method.invoke(target, objects);
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
接着我们来创建一下代理类:
java
public class CglibProxyMain {
public static void main(String[] args) {
SubjectImpl subject = new SubjectImpl();
//增强器,一会就需要依靠增强器来为我们生成动态代理对象
Enhancer enhancer = new Enhancer();
//直接选择我们需要代理的类型,直接不需要接口或是抽象类,SuperClass作为代理类的父类存在,这样我们就可以按照指定类型的方式去操作代理类了
enhancer.setSuperclass(SubjectImpl.class);
//设定我们刚刚编写好的代理逻辑
enhancer.setCallback(new TestCglibProxy(subject));
//直接创建代理类
SubjectImpl proxy = (SubjectImpl) enhancer.create();
//调用代理类的test方法
proxy.test();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
外观模式
外观模式充分体现了迪米特法则。 可能我们的整个项目有很多个子系统, 但是我们可以在这些子系统的上面加一个门面(Facade)当我们外部需要与各个子系统交互时, 无需再去直接使用各个子系统,而是与门面进行交互,再由门面与后面的各个子系统操作, 这样,我们以后需要办什么事情,就统一找门面就行了。 这样的好处是,首先肯定方便了代码的编写,统一找门面就行,不需要去详细了解子系统。 并且当子系统需要修改时,也只需要修改门面中的逻辑,不需要大面积的变动,遵循迪米特法则尽可能少的交互。
java
class SubSystemA {
public void test1() {
System.out.println("排队");
}
}
class SubSystemB {
public void test1() {
System.out.println("领证");
}
}
class SubSystemC {
public void test1() {
System.out.println("结婚");
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
现在三个系统太复杂了,我们添加一个门面:
java
class Facade {
SubSystemA a = new SubSystemA();
SubSystemB b = new SubSystemB();
SubSystemC c = new SubSystemC();
public void marry() { //红白喜事一条龙服务
a.test1();
b.test1();
c.test1();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
现在我们只需要一个门面就能直接把事情办完了:
java
public class FacadeModeMain {
public static void main(String[] args) {
Facade facade = new Facade();
facade.marry();
}
}
1
2
3
4
5
6
7
2
3
4
5
6
7
通过使用外观模式,我们就大大降低了类与类直接的关联程度,并且简化了流程
享元模式
我们可以将那些重复出现的内容作为共享部分取出, 这样当我们拥有大量对象时,我们把其中共同的部分抽取出来, 由于提取的部分是多个对象共享只有一份,那么就可以减轻内存的压力。
比如现在我们有两个服务,但是他们都需要使用数据库工具类来操作,实际上这个工具类没必要创建多个, 我们这时就可以使用享元模式,让数据库工具类作为享元类,通过享元工厂来提供一个共享的数据库工具类:
java
class DBUtil {
public void selectDB() {
System.out.println("我是数据库操作...");
}
}
class DBUtilFactory {
//享元对象被存放在工厂中
private static final DBUtil UTIL = new DBUtil();
//获取享元对象
public static DBUtil getFlyweight() {
return UTIL;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2
3
4
5
6
7
8
9
10
11
12
13
14
15
当我们需要使用享元对象时,直接找享元工厂要就行了:
java
//用户服务
class UserService {
public void service() {
//通过享元工厂拿到DBUtil对象
DBUtil util = DBUtilFactory.getFlyweight();
//该干嘛干嘛
util.selectDB();
}
}
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
实际上 String类,也在使用享元模式进行优化,比如下面的代码:
java
public class String {
public static void main(String[] args) {
String str1 = "abcd";
String str2 = "abcd";
String str3 = "ab" + "cd";
System.out.println(str1 == str2);
System.out.println(str1 == str3); //猜猜这三个对象是不是都是同一个?
}
}
1
2
3
4
5
6
7
8
9
2
3
4
5
6
7
8
9