这是我参与8月更文挑战的第1天,活动详情查看:8月更文挑战
介绍
访问者模式
的目的是增加一个新的操作
而不改变
对现有对象结构
的修改
。
试想一下,有一个由组件组成的复合对象,对象的结构是固定的.我们不能去改变它,就意味着我们不能向结构添加新的元素。
现在,我们如何在不修改现有类的情况下向我们的代码添加新功能?
访问者模式
或许能帮我们解决这个问题.
举个栗子
现在处于疫情期间,假若有一个系统,存储了我们的个人信息.方便我们出行的时候提供健康码、第几针疫苗以及个人最基本的手机号,身份证等信息.
现在存在两个部门,各部门分不同的任务,如下
- 医院(姓名,身份证等…)
- 通行检查处(检查疫苗接种等…)
使用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图:
如何学习访问者模式
刚学访问者模式
的时候,我也很是费解,因为在下的理解能力有限
那怎么办呢,敲代码,多找一些访问者模式
的博客,看上面的例子.
不要复制粘贴
, 不然你很容易错过细节
.细节很细,你忍一下
.
同时,不要去害怕它.一天不行就两天,每天抽出那么点时间来
,去敲一个访问者模式的例子
.
然后尝试着去自己写一个例子,再去写一些总结.
最后,在看一些概念性的东西.这也是为啥要把说明放后面的原因
.
很多博客概念直接劝退,不如代码来的实在.
八股文概念
我们要做的就是向结构的每个元素添加一个接受访问者类的函数.
这样我们的组件将允许访问者实现访问它们并对该元素执行任何所需的操作.
因为我们不会修改代码,但我们仍然可以通过提供新的访问者实现来扩展功能.
例如:在上面的举例中,我们可以在不更改现有类的情况下,增加一个处理黄码的逻辑.
今天的文章Java进阶-设计模式-访问者模式分享到此就结束了,感谢您的阅读。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
如需转载请保留出处:https://bianchenghao.cn/21351.html