1700472360
由于职责分开,继续增加对数据的操作是非常快捷的,例如现在要增加一份给大老板的报表,这份报表格式又有所不同,直接在Visitor中增加一个方法,传递数据后进行整理打印。
1700472361
1700472362
❑灵活性非常高
1700472363
1700472364
例如,数据汇总,就以刚刚我们说的Employee的例子,如果我现在要统计所有员工的工资之和,怎么计算?把所有人的工资for循环加一遍?是个办法,那我再提个问题,员工工资×1.2,部门经理×1.4,总经理×1.8,然后把这些工资加起来,你怎么处理?1.2,1.4,1.8是什么?不是吧?!你没看到领导不论什么时候都比你拿得多,工资奖金就不说了,就是过节发个慰问券也比你多,就是这个系数在作祟。我们继续说你想怎么统计?使用for循环,然后使用instanceof来判断是员工还是经理?这可以解决,但不是个好办法,好办法是通过访问者模式来实现,把数据扔给访问者,由访问者来进行统计计算。
1700472365
1700472366
1700472367
1700472368
1700472369
设计模式之禅 25.3.2 访问者模式的缺点
1700472370
1700472371
❑具体元素对访问者公布细节
1700472372
1700472373
访问者要访问一个类就必然要求这个类公布一些方法和数据,也就是说访问者关注了其他类的内部细节,这是迪米特法则所不建议的。
1700472374
1700472375
❑具体元素变更比较困难
1700472376
1700472377
具体元素角色的增加、删除、修改都是比较困难的,就上面那个例子,你想想,你要是想增加一个成员变量,如年龄age,Visitor就需要修改,如果Visitor是一个还好办,多个呢?业务逻辑再复杂点呢?
1700472378
1700472379
❑违背了依赖倒置转原则
1700472380
1700472381
访问者依赖的是具体元素,而不是抽象元素,这破坏了依赖倒置原则,特别是在面向对象的编程中,抛弃了对接口的依赖,而直接依赖实现类,扩展比较难。
1700472382
1700472383
1700472384
1700472385
1700472386
设计模式之禅 25.3.3 访问者模式的使用场景
1700472387
1700472388
❑一个对象结构包含很多类对象,它们有不同的接口,而你想对这些对象实施一些依赖于其具体类的操作,也就说是用迭代器模式已经不能胜任的情景。
1700472389
1700472390
❑需要对一个对象结构中的对象进行很多不同并且不相关的操作,而你想避免让这些操作“污染”这些对象的类。
1700472391
1700472392
总结一下,在这种地方你一定要考虑使用访问者模式:业务规则要求遍历多个不同的对象。这本身也是访问者模式出发点,迭代器模式只能访问同类或同接口的数据(当然了,如果你使用instanceof,那么能访问所有的数据,这没有争论),而访问者模式是对迭代器模式的扩充,可以遍历不同的对象,然后执行不同的操作,也就是针对访问的对象不同,执行不同的操作。访问者模式还有一个用途,就是充当拦截器(Interceptor)角色,这个我们将在混编模式中讲解。
1700472393
1700472394
1700472395
1700472396
1700472398
设计模式之禅 25.4 访问者模式的扩展
1700472399
1700472400
访问者模式是经常用到的模式,虽然你不注意,有可能你起的名字也不是什么Visitor,但是它确实是非常容易使用到的,在这里我提出两个扩展的功能供大家参考。
1700472401
1700472402
25.4.1 统计功能
1700472403
1700472404
在例子中我们也提到访问者的统计功能,汇总和报表是金融类企业非常常用的功能,基本上都是一堆的计算公式,然后出一个报表,很多项目采用了数据库的存储过程来实现,我不是很推荐这种方式,除非海量数据处理,一个晚上要批处理上亿、几十亿条的数据,除了存储过程来处理还没有其他办法,你要是用应用服务器来处理,连接数据库的网络就是处于100%占用状态,一个晚上也未必能处理完这批数据!除了这种海量数据外,我建议数据统计和报表的批处理通过访问者模式来处理会比较简单。好,那我们来统计一下公司人员的工资总额,先看类图,如图25-6所示。
1700472405
1700472406
1700472407
1700472408
1700472409
图25-6 统计功能的访问者模式
[
上一页 ]
[ :1.70047236e+09 ]
[
下一页 ]