Java进阶-设计模式-访问者模式

124 篇文章 1 订阅

介绍

访问者模式的目的是增加一个新的操作不改变现有对象结构修改
试想一下,有一个由组件组成的复合对象,对象的结构是固定的.我们不能去改变它,就意味着我们不能向结构添加新的元素。
现在,我们如何在不修改现有类的情况下向我们的代码添加新功能?
访问者模式或许能帮我们解决这个问题.

举个栗子

现在处于疫情期间,假若有一个系统,存储了我们的个人信息.方便我们出行的时候提供健康码、第几针疫苗以及个人最基本的手机号,身份证等信息.
现在存在两个部门,各部门分不同的任务,如下

  • 医院(姓名,身份证等...)
  • 通行检查处(检查疫苗接种等...)

使用Java代码实现,具体如下:
定义被访问者接口:

public interface Person {
    void accept(Department department);
}
复制代码

定义红绿码成员:

public class RedCodePerson implements Person {
    private String userName;

    private String userPhone;

    private String cardNo;

    private String address;

    @Override
    public void accept(Department department) {
        department.visit(this);
    }

    public RedCodePerson(String userName, String userPhone, String cardNo, String address) {
        this.userName = userName;
        this.userPhone = userPhone;
        this.cardNo = cardNo;
        this.address = address;
    }

    public String getUserName() {
        return userName;
    }

    public String getUserPhone() {
        return userPhone;
    }

    public String getCardNo() {
        return cardNo;
    }

    public String getAddress() {
        return address;
    }

}
复制代码

定义绿码成员:

package com.home.test.visitor.userInfo;

public class GreenCodePerson implements Person{

    private String userName;

    private String userPhone;

    private String cardNo;

    private String address;

    // 疫苗
    private String vaccine;

    @Override
    public void accept(Department department) {
        department.visit(this);
    }

    public GreenCodePerson(String userName, String userPhone, String cardNo, String address, String vaccine) {
        this.userName = userName;
        this.userPhone = userPhone;
        this.cardNo = cardNo;
        this.address = address;
        this.vaccine = vaccine;
    }

    public String getUserName() {
        return userName;
    }

    public String getUserPhone() {
        return userPhone;
    }

    public String getCardNo() {
        return cardNo;
    }

    public String getAddress() {
        return address;
    }

    public String getVaccine() {
        return vaccine;
    }
}
复制代码

定义访问者部门接口:

public interface Department {

    void visit(RedCodePerson person);

    void visit(GreenCodePerson person);

}
复制代码

定义医院部门访问实现类:

package com.home.test.visitor.userInfo;

public class DocDepartment implements Department {
    @Override
    public void visit(RedCodePerson person) {
        System.out.printf("红码人员:%s,手机号:%s,疫苗接种情况:%s,开始隔离!%n",
                person.getUserName(),
                person.getUserPhone(),
                person.getAddress());
    }

    @Override
    public void visit(GreenCodePerson person) {
        System.out.printf("绿码人员:%s,手机号:%s,疫苗接种情况:%s,放回家.%n",
                person.getUserName(),
                person.getUserPhone(),
                person.getVaccine());
    }
}
复制代码

定义乡镇访问实现类:

package com.home.test.visitor.userInfo;

/**
 * 乡镇获取权限: 姓名、手机号、接种情况、地址、身份证号
 */
public class VillageDepartment implements Department {

    @Override
    public void visit(RedCodePerson person) {
        System.out.printf("红码人员:%s,手机号:%s,身份证号:%s,地址:%s,请及时排查密接人员!%n",
                person.getUserName(),
                person.getUserPhone(),
                person.getCardNo(),
                person.getAddress());
    }

    @Override
    public void visit(GreenCodePerson person) {
        System.out.printf("绿码人员:%s,手机号:%s,身份证号:%s,疫苗接种情况:%s,",
                person.getUserName(),
                person.getUserPhone(),
                person.getCardNo(),
                person.getVaccine());
        if (("未接种").equals(person.getVaccine())) {
            System.out.println("请及时通知该人员接种疫苗");
        } else {
            System.out.println("请提醒多喝热水.");
        }
    }

}
复制代码

定义测试Demo:

package com.home.test.visitor.userInfo;

import java.util.ArrayList;
import java.util.List;

public class ClientDemo {

    public static void main(String[] args) {
        List<Person> personList = new ArrayList<Person>() {{
            add(new GreenCodePerson("Job", "13188880000", "866111", "New Work", "未接种"));
            add(new GreenCodePerson("Job2", "13188889999", "866112", "New Work", "未接种"));
            add(new GreenCodePerson("Job3", "13200000000", "866113", "New Work", "已接种"));
            add(new RedCodePerson("李富贵", "13200000001", "866114", "New Work"));
        }};
        System.out.println("------------------------医院权限查看------------------------");
        DocDepartment docDepartment = new DocDepartment();
        personList.forEach(person -> person.accept(docDepartment));
        System.out.println("------------------------乡镇部门权限查看------------------------");
        VillageDepartment villageDepartment = new VillageDepartment();
        personList.forEach(person -> person.accept(villageDepartment));
    }
}
复制代码

输出结果:

------------------------医院权限查看------------------------
绿码人员:Job,手机号:13188880000,疫苗接种情况:未接种,放回家.
绿码人员:Job2,手机号:13188889999,疫苗接种情况:未接种,放回家.
绿码人员:Job3,手机号:13200000000,疫苗接种情况:已接种,放回家.
红码人员:李富贵,手机号:13200000001,疫苗接种情况:New Work,开始隔离!
------------------------乡镇部门权限查看------------------------
绿码人员:Job,手机号:13188880000,身份证号:866111,疫苗接种情况:未接种,请及时通知该人员接种疫苗
绿码人员:Job2,手机号:13188889999,身份证号:866112,疫苗接种情况:未接种,请及时通知该人员接种疫苗
绿码人员:Job3,手机号:13200000000,身份证号:866113,疫苗接种情况:已接种,请提醒多喝热水.
红码人员:李富贵,手机号:13200000001,身份证号:866114,地址:New Work,请及时排查密接人员!
复制代码

说明

访问者模式属于行为设计模式的一种,当我们要对一组相似类型的对象执行操作的时候使用它。
访问者模式的帮助下,我们可以将操作逻辑从对象移动到另一个类。

访问者模式由两部分组成:
一个名为 Visit() 的方法,由访问者实现,并为数据结构中的每个元素调用
提供接受访问者的 Accept() 方法的可访问类

如下UML图:

Blank diagram.png

如何学习访问者模式

刚学访问者模式的时候,我也很是费解,因为在下的理解能力有限

我擦泪.jpg

那怎么办呢,敲代码,多找一些访问者模式的博客,看上面的例子.
不要复制粘贴, 不然你很容易错过细节.细节很细,你忍一下.
同时,不要去害怕它.一天不行就两天,每天抽出那么点时间来,去敲一个访问者模式的例子.
然后尝试着去自己写一个例子,再去写一些总结.
最后,在看一些概念性的东西.这也是为啥要把说明放后面的原因.
很多博客概念直接劝退,不如代码来的实在.

八股文概念

我们要做的就是向结构的每个元素添加一个接受访问者类的函数.
这样我们的组件将允许访问者实现访问它们并对该元素执行任何所需的操作.
因为我们不会修改代码,但我们仍然可以通过提供新的访问者实现来扩展功能.
例如:在上面的举例中,我们可以在不更改现有类的情况下,增加一个处理黄码的逻辑.


作者:铅华56
链接:https://juejin.cn/post/6995462139947253773
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值