Java設(shè)計(jì)模式之訪問者模式使用場景及代碼示例
Java設(shè)計(jì)模式訪問者模式
模式概念
訪問者模式表示一個(gè)作用于某對(duì)象結(jié)構(gòu)中的各元素的操作,它使你可以在不改變各元素類的前提下定義作用于這些元素的新操作。訪問者模式適用于數(shù)據(jù)結(jié)構(gòu)相對(duì)穩(wěn)定算法又易變化的系統(tǒng),若系統(tǒng)數(shù)據(jù)結(jié)構(gòu)對(duì)象易于變化,則不適合使用訪問者模式。訪問者模式的優(yōu)點(diǎn)是增加操作很容易,因?yàn)樵黾硬僮饕馕吨黾有碌脑L問者。
Visitor應(yīng)用場景
一定會(huì)有的疑問:visitor和iterator的區(qū)別:
visitor可以訪問不同的對(duì)象(只需要在Element定義對(duì)應(yīng)的accept),但是Iterator只能訪問相同的對(duì)象,最起碼要有相同的接口
iterator是不依賴具體實(shí)現(xiàn)的,而visitor是依賴具體實(shí)現(xiàn)的,因?yàn)閂isitor會(huì)根據(jù)訪問的具體的對(duì)象來采取對(duì)應(yīng)的操作,而iterator最多只是基于相同的接口的泛化實(shí)現(xiàn)。
iterator訪問的數(shù)據(jù)結(jié)構(gòu)的操作和數(shù)據(jù)并未分離,所以拓展功能起來需要修改,違反了開閉原則和單一職責(zé)原則。但是因?yàn)樵L問者依賴具體實(shí)現(xiàn),而不是依賴抽象,所以違反了依賴倒置原則
優(yōu)缺點(diǎn)決定的應(yīng)用場景
符合單一職責(zé)原則,功能上具有良好的拓展性,但是因?yàn)橐蕾嚲唧w實(shí)現(xiàn)違背了具體實(shí)現(xiàn),所以為類的修改帶了麻煩。
具有優(yōu)良的拓展性,只需要實(shí)現(xiàn)新的Visitor來滿足新的訪問要求。因?yàn)閿?shù)據(jù)和操作的分離,防止了添加新的操作污染原來的數(shù)據(jù)結(jié)構(gòu)。
綜上
訪問者是一種集中規(guī)整模式,特別適合用于大規(guī)模重構(gòu)的項(xiàng)目,在這一個(gè)階段的需求已經(jīng)非常清晰,原系統(tǒng)的功能點(diǎn)也已經(jīng)明確,通過訪問者模式可以很容易把一些功能進(jìn)行梳理,達(dá)到最終目的功能集中化
模式結(jié)構(gòu)
1)Visitor 抽象訪問者角色:
為該對(duì)象結(jié)構(gòu)中具體元素角色聲明一個(gè)訪問操作接口。該操作接口的名字和參數(shù)標(biāo)識(shí)了發(fā)送訪問請(qǐng)求給具體訪問者的具體元素角色,這樣訪問者就可以通過該元素角色的特定接口直接訪問它。
2)ConcreteVisitor具體訪問者角色:
實(shí)現(xiàn)Visitor聲明的接口。
3)Element抽象受訪元素:
定義一個(gè)接受訪問操作(accept()),它以一個(gè)訪問者(Visitor)作為參數(shù)。
4)ConcreteElement 具體受訪元素:
實(shí)現(xiàn)了抽象元素(Element)所定義的接受操作接口。
5)ObjectStructure 結(jié)構(gòu)對(duì)象角色:
這是使用訪問者模式必備的角色。它具備以下特性:能枚舉它的元素;可以提供一個(gè)高層接口以允許訪問者訪問它的元素;如有需要,可以設(shè)計(jì)成一個(gè)復(fù)合對(duì)象或者一個(gè)聚集(如一個(gè)列表或無序集合)。
Demo
抽象訪問者角色:
public interface IVisitor { public void accept(Feman feman); public void accept(Man man); }
具體訪問角色:
public class Visitor implements IVisitor { public void accept(Feman feman) { System.out.println(feman.getSex() + ":執(zhí)行相關(guān)操作"); } public void accept(Man man) { System.out.println(man.getSex() + ":執(zhí)行相關(guān)操作"); } }
(注)Visitor中設(shè)置了同樣的名稱的方法且方法傳參為實(shí)現(xiàn)同一接口的不同對(duì)象,即受訪者元素。
抽象受訪元素:
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) { }; }
具體受訪元素:
public class Man extends Person { public Man() { this.setSex("男"); } @Override public void accept(Visitor visitor) { visitor.accept(this); } }
public class Feman extends Person { public Feman() { this.setSex("女"); } @Override public void accept(Visitor visitor){ visitor.accept(this); } }
結(jié)構(gòu)對(duì)象角色:
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; } }
執(zhí)行過程:
Visitor visitor = new Visitor(); List<person> list = ObjectStruture.getList(); for (Person e : list) { e.accept(visitor); }
執(zhí)行結(jié)果:
男:執(zhí)行相關(guān)操作 女:執(zhí)行相關(guān)操作 女:執(zhí)行相關(guān)操作
下面是一個(gè)完整的代碼示例:
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); } }
運(yùn)行結(jié)果:
GladiolusVisitor access GladiolusConcreteElement GladiolusVisitor access ChrysanthemumConreteElement ChrysanthemumVisitor access GladiolusConcreteElement ChrysanthemumVisitor access ChrysanthemumConreteElement
總結(jié)
以上就是本文關(guān)于Java設(shè)計(jì)模式之訪問者模式使用場景及代碼示例的全部內(nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:Java編程—在測試中考慮多態(tài)、Java實(shí)現(xiàn)微信公眾平臺(tái)朋友圈分享功能詳細(xì)代碼、Java編程BigDecimal用法實(shí)例分享等,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持。
相關(guān)文章
Java實(shí)戰(zhàn)網(wǎng)上電子書城的實(shí)現(xiàn)流程
讀萬卷書不如行萬里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+SSM+JSP+maven+Mysql實(shí)現(xiàn)一個(gè)網(wǎng)上電子書城,大家可以在過程中查缺補(bǔ)漏,提升水平2022-01-01Java @Value("${xxx}")取properties時(shí)中文亂碼的解決
這篇文章主要介紹了Java @Value("${xxx}")取properties時(shí)中文亂碼的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07Springboot 如何指定獲取自己寫的配置properties文件的值
這篇文章主要介紹了Springboot 如何指定獲取自己寫的配置properties文件的值,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07Java中使用Hutool的DsFactory操作多數(shù)據(jù)源的實(shí)現(xiàn)
在Java開發(fā)中,管理多個(gè)數(shù)據(jù)源是一項(xiàng)常見需求,Hutool作為一個(gè)全能的Java工具類庫,提供了DsFactory工具,幫助開發(fā)者便捷地操作多數(shù)據(jù)源,感興趣的可以了解一下2024-09-09Mybatis和Mybatis-Plus時(shí)間范圍查詢方式
這篇文章主要介紹了Mybatis和Mybatis-Plus時(shí)間范圍查詢方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-08-08MybatisPlus查詢數(shù)據(jù)日期格式化問題解決方法
MyBatisPlus是MyBatis的增強(qiáng)工具,支持常規(guī)的CRUD操作以及復(fù)雜的聯(lián)表查詢等功能,這篇文章主要給大家介紹了關(guān)于MybatisPlus查詢數(shù)據(jù)日期格式化問題的解決方法,需要的朋友可以參考下2023-10-10