apmserv 设置网站目录,深圳专门网站建设,做 网站 要专线吗,华强北商城官网入口访问者模式难以实现#xff0c;且应用该模式可能会导致代码可读性变差#xff0c;可维护性变差#xff0c;除非必要#xff0c;不建议使用#xff1b;
1.访问者模式定义 允许在运行时将一个或多个操作应用于一组对象#xff0c;将操作与对象结构分离#xff1b; 访问者… 访问者模式难以实现且应用该模式可能会导致代码可读性变差可维护性变差除非必要不建议使用
1.访问者模式定义 允许在运行时将一个或多个操作应用于一组对象将操作与对象结构分离 访问者模式主要解决的是数据与算法的耦合问题尤其是在数据结构比较稳定而算法多变的情况下为了不污染数据本身访问者会将多种算法独立归档并在访问数据时根据数据类型自动切换到对应的算法实现数据的自动响应机制并确保算法的自由扩展
1.1 访问者模式优缺点
优点
扩展性好在不修改对象结构中元素的情况下为对象结构中的元素添加新的功能复用性好通过访问者来定义整个对象结构通用的功能从而提高复用程度分离无关行为把相关的行为封装在一起构成一个访问者这样每一个访问者的功能都比较单一
缺点
对象结构变化很困难在访问者模式中没增加一个新的元素类都要在每一个具体访问者类中增加相应的具体操作这违背了开闭原则违反了依赖倒置原则访问者模式依赖了具体类而没有依赖抽象类
1.2 访问者模式的使用场景
当对象的数据结构相对稳定而操作却经常变化的时候。 比如上面例子中路由器本身的内部构造也就是数据结构不会怎么变化但是在不同操作系统下的操作可能会经常变化比如发送数据、接收数据等。需要将数据结构与不常用的操作进行分离的时候。 比如扫描文件内容这个动作通常不是文件常用的操作但是对于文件夹和文件来说和数据结构本身没有太大关系树形结构的遍历操作扫描是一个额外的动作如果给每个文件都添加一个扫描操作会太过于重复这时采用访问者模式是非常合适的能够很好分离文件自身的遍历操作和外部的扫描操作。需要在运行时动态决定使用哪些对象和方法的时候。 比如对于监控系统来说很多时候需要监控运行时的程序状态但大多数时候又无法预知对象编译时的状态和参数这时使用访问者模式就可以动态增加监控行为。
2.访问者模式原理 抽象访问者Visitor可以是接口或抽象类定义了一系列操作方法用来处理所有数据元素通常为同名的访问方法并以数据元素作为入参来确定哪个重载方法被调用具体访问者ConcreteVisitor实现了抽象访问者可以有多个每个访问者都需要实现所有数据元素类型的访问重载发方法抽象元素类Element被访问的数据元素接口定义了一个接受访问者的方法每个元素都要可以被访问者访问具体元素类ConcreteElement具体数据元素实现类提供接受访问方法的具体实现而这个具体的实现通常情况下是使用访问者提供的访问该元素类的方法对象结构类ObjectStructure包含所有可能被访问的数据对象的容器可以提供数据对象的迭代功能可以是任意类型的数据结构客户端使用容器并初始化其中各类数据元素并选择合适的访问者处理容器中的所有数据对象
3.访问者模式的实现
【实例】 以超市购物为例假设超市中的两类商品糖果、酒水进行售卖。我们可以忽略每种商品的计价方法因为最终结账时由收银员统一集中处理在商品类中添加计价方法是不合理的设计
【代码】 访问者接口
public interface Visitor {public void visit(Wine wine); //酒类重载方法public void visit(Candy candy); //糖果重载方法
} 接待者接口
public interface Acceptable {//接收所有的Visitor访问者的子类实现类public void accept(Visitor visitor);
} 糖果类酒类类似
public class Candy extends Product implements Acceptable{public Candy(String name, LocalDate producedDate, double price) {super(name, producedDate, price);}Overridepublic void accept(Visitor visitor) {//accept实现方法中调用访问者并将自己 this 传回。this是一个明确的身份,不存在任何泛型visitor.visit(this);}
} 具体访问者——折扣计价访问者
public class DiscountVisitor implements Visitor {private LocalDate billDate;public DiscountVisitor(LocalDate billDate) {this.billDate billDate;System.out.println(结算日期: billDate);}Overridepublic void visit(Candy candy) {System.out.println(糖果: candy.getName());//获取产品生产天数long days billDate.toEpochDay() - candy.getProducedDate().toEpochDay();if(days 180){System.out.println(超过半年的糖果,请勿食用!);}else{double rate 0.9;double discountPrice candy.getPrice() * rate;System.out.println(糖果打折后的价格NumberFormat.getCurrencyInstance().format(discountPrice));}}Overridepublic void visit(Wine wine) {System.out.println(酒类: wine.getName(),无折扣价格!);System.out.println(原价: NumberFormat.getCurrencyInstance().format(wine.getPrice()));}
} 客户端
public class Client {public static void main(String[] args) {//模拟添加多个商品的操作ListAcceptable products Arrays.asList(new Candy(金丝猴奶糖,LocalDate.of(2022,6,10),10.00),new Wine(衡水老白干,LocalDate.of(2020,6,10),100.00));Visitor visitor new DiscountVisitor(LocalDate.of(2022,10,17));for (Acceptable product : products) {product.accept(visitor);}}
}