Java设计模式之访问者模式使用场景及代码示例

时间:2022-03-24 23:25:13

Java设计模式访问者模式

模式概念

访问者模式表示一个作用于某对象结构中的各元素的操作,它使你可以在不改变各元素类的前提下定义作用于这些元素的新操作。访问者模式适用于数据结构相对稳定算法又易变化的系统,若系统数据结构对象易于变化,则不适合使用访问者模式。访问者模式的优点是增加操作很容易,因为增加操作意味着增加新的访问者。

Visitor应用场景

一定会有的疑问:visitor和iterator的区别:

visitor可以访问不同的对象(只需要在Element定义对应的accept),但是Iterator只能访问相同的对象,最起码要有相同的接口
iterator是不依赖具体实现的,而visitor是依赖具体实现的,因为Visitor会根据访问的具体的对象来采取对应的操作,而iterator最多只是基于相同的接口的泛化实现。

iterator访问的数据结构的操作和数据并未分离,所以拓展功能起来需要修改,违反了开闭原则和单一职责原则。但是因为访问者依赖具体实现,而不是依赖抽象,所以违反了依赖倒置原则

优缺点决定的应用场景

符合单一职责原则,功能上具有良好的拓展性,但是因为依赖具体实现违背了具体实现,所以为类的修改带了麻烦。

具有优良的拓展性,只需要实现新的Visitor来满足新的访问要求。因为数据和操作的分离,防止了添加新的操作污染原来的数据结构。

综上

访问者是一种集中规整模式,特别适合用于大规模重构的项目,在这一个阶段的需求已经非常清晰,原系统的功能点也已经明确,通过访问者模式可以很容易把一些功能进行梳理,达到最终目的功能集中化

模式结构

1)Visitor 抽象访问者角色:
为该对象结构中具体元素角色声明一个访问操作接口。该操作接口的名字和参数标识了发送访问请求给具体访问者的具体元素角色,这样访问者就可以通过该元素角色的特定接口直接访问它。

2)ConcreteVisitor具体访问者角色:
实现Visitor声明的接口。

3)Element抽象受访元素:
定义一个接受访问操作(accept()),它以一个访问者(Visitor)作为参数。

4)ConcreteElement 具体受访元素:
实现了抽象元素(Element)所定义的接受操作接口。

5)ObjectStructure 结构对象角色:
这是使用访问者模式必备的角色。它具备以下特性:能枚举它的元素;可以提供一个高层接口以允许访问者访问它的元素;如有需要,可以设计成一个复合对象或者一个聚集(如一个列表或无序集合)。

Demo

抽象访问者角色:

?
1
2
3
4
5
public interface IVisitor {
 public void accept(Feman feman);
 
 public void accept(Man man);
}

具体访问角色:

?
1
2
3
4
5
6
7
8
9
public class Visitor implements IVisitor {
 public void accept(Feman feman) {
  System.out.println(feman.getSex() + ":执行相关操作");
 }
 
 public void accept(Man man) {
  System.out.println(man.getSex() + ":执行相关操作");
 }
}

(注)Visitor中设置了同样的名称的方法且方法传参为实现同一接口的不同对象,即受访者元素。

抽象受访元素:

?
1
2
3
4
5
6
7
8
9
10
11
public abstract class Person {
 private String sex;
 public String getSex() {
  return sex;
 }
 public void setSex(String sex) {
  this.sex = sex;
 }
 public void accept(Visitor visitor) {
 };
}

具体受访元素:

?
1
2
3
4
5
6
7
8
9
10
public class Man extends Person {
 public Man() {
  this.setSex("男");
 }
 
 @Override
 public void accept(Visitor visitor) {
  visitor.accept(this);
 }
}
?
1
2
3
4
5
6
7
8
9
10
public class Feman extends Person {
 public Feman() {
  this.setSex("女");
 }
  
 @Override
 public void accept(Visitor visitor){
  visitor.accept(this);
 }
}

结构对象角色:

?
1
2
3
4
5
6
7
8
9
public class ObjectStruture {
 public static List<person> getList() {
  List<person> list = new ArrayList<person>();
  list.add(new Man());
  list.add(new Feman());
  list.add(new Feman());
  return list;
 }
}

执行过程:

?
1
2
3
4
5
Visitor visitor = new Visitor();
List<person> list = ObjectStruture.getList();
for (Person e : list) {
 e.accept(visitor);
}

执行结果:

?
1
2
3
男:执行相关操作
女:执行相关操作
女:执行相关操作

下面是一个完整的代码示例:

?
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
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
public interface Visitor
{
    public void visit(GladiolusConcreteElement gladiolus);
    public void visit(ChrysanthemumConreteElement chrysanthemum);
}
public interface FlowerElement
{
    public void accept(Visitor visitor);
}
public class GladiolusConcreteElement implements FlowerElement
{
    @Override
     public void accept(final Visitor visitor)
     {
        visitor.visit(this);
    }
}
public class ChrysanthemumConreteElement implements FlowerElement
{
    @Override
     public void accept(final Visitor visitor)
     {
        visitor.visit(this);
    }
}
public class GladiolusVisitor implements Visitor
{
    @Override
     public void visit(final GladiolusConcreteElement gladiolus)
     {
        System.out.println(this.getClass().getSimpleName() + " access " + gladiolus.getClass().getSimpleName());
    }
    @Override
     public void visit(final ChrysanthemumConreteElement chrysanthemum)
     {
        System.out.println(this.getClass().getSimpleName() + " access " + chrysanthemum.getClass().getSimpleName());
    }
}
public class ChrysanthemumConreteElement implements FlowerElement
{
    @Override
     public void accept(final Visitor visitor)
     {
        visitor.visit(this);
    }
}
public class ObjectStructure
{
    private final List<FlowerElement> elements = new ArrayList<FlowerElement>();
    public void addElement(final FlowerElement e)
     {
        elements.add(e);
    }
    public void removeElement(final FlowerElement e)
     {
        elements.remove(e);
    }
    public void accept(final Visitor visitor)
     {
        for (final FlowerElement e : elements)
          {
            e.accept(visitor);
        }
    }
}
public class Client
{
    public static void main(final String[] args)
     {
        final ObjectStructure os = new ObjectStructure();
        os.addElement(new GladiolusConcreteElement());
        os.addElement(new ChrysanthemumConreteElement());
        final GladiolusVisitor gVisitor = new GladiolusVisitor();
        final ChrysanthemumVisitor chVisitor = new ChrysanthemumVisitor();
        os.accept(gVisitor);
        os.accept(chVisitor);
    }
}

运行结果:

?
1
2
3
4
GladiolusVisitor access GladiolusConcreteElement
GladiolusVisitor access ChrysanthemumConreteElement
ChrysanthemumVisitor access GladiolusConcreteElement
ChrysanthemumVisitor access ChrysanthemumConreteElement

总结

以上就是本文关于Java设计模式之访问者模式使用场景及代码示例的全部内容,希望对大家有所帮助。如有不足之处,欢迎留言指出。感谢朋友们对本站的支持。

原文链接:https://www.2cto.com/kf/201612/582504.html