访问者模式有点小复杂,而且需要在实际的时候提前留好后门,下面通过简单的例子来介绍一下。场景就是邮件发送,通知客户怎么了怎么了,后面可能需要短信或者app也推。推之前或推送之后干点别的。
public abstract class BaseJob { public abstract void Execute();public abstract void Accept(Monitoring monitoring); }
这里的Execute方法就是正常操作,Accept就是预留的方法,通过它可以访问BaseJob 的实现,并且在不改动BaseJob 的情况下。那么Monitoring这个对象是什么呢?
public abstract class Monitoring { public abstract void Monitor(SendEmail sendEmail); public abstract void Monitor(SendMessage sendMessage); public abstract void Monitor(SendOther endOther); }
所谓访问者Monitoring肯定是 BaseJob 的访问者,所以SendEmail 、SendMessage 、SendOther 都是BaseJob 的实现
public class SendEmail : BaseJob { public override void Accept(Monitoring monitoring) { monitoring.Monitor(this); } public override void Execute() { Debug.WriteLine("邮件发送"); } } public class SendMessage : BaseJob { public override void Accept(Monitoring monitoring) { monitoring.Monitor(this); } public override void Execute() { Debug.WriteLine("短信发送"); } } public class SendOther : BaseJob { public override void Accept(Monitoring monitoring) { monitoring.Monitor(this); } public override void Execute() { Debug.WriteLine("App发送"); } }
被访问的对象实现了,这个访问者也先给实现了吧。
public class AutoMonitor: Monitoring { public override void Monitor(SendEmail sendEmail) { Debug.WriteLine("邮件发送前"); sendEmail.Execute(); Debug.WriteLine("邮件发送后"); } public override void Monitor(SendMessage sendMessage) { Debug.WriteLine("短信发送前"); sendMessage.Execute(); Debug.WriteLine("短信发送后"); } public override void Monitor(SendOther endOther) { Debug.WriteLine("app发送前"); endOther.Execute(); Debug.WriteLine("app发送后"); } }
下面就是怎么去调用的问题了。通过一个director类去包装一下访问者和被访问者。
public class Director { private Monitoring _monitoring; public Director(Monitoring monitoring) { _monitoring = monitoring; } public void Start(BaseJob baseJob) { baseJob.Accept(_monitoring); } }
director的代码有点反转再反转的味道,Start方法是我们访问的入口,通过传一个被访问者对象,调用被访问者的Accept的方法,被访问者Accept又需要一个访问者的对象,这里就通过构造函数传进来的,目前为止都是抽象的,还未具体指定实现。
执行的流程就是start=>accept=》moitor,director入口,到basejob的accept拿到访问者monitor,mointor的实现moitor又可以拿到完整的被访问者basejob,这样就可以操纵被访问者了,而且可以在访问者前后添加任何扩展自己的操作。
有点绕,但是这个也实现了aop,跟装饰器效果也差不多了。
下面就是客户端的调用了,这里需要访问SendEmail 、SendMessage 、SendOther 都可以灵活的替换掉。
public void TestV() { Monitoring monitoring = new AutoMonitor(); Director director = new Director(monitoring); director.Start(new SendEmail()); }
写到这里差不多了,留个彩蛋自己体会:
public class Director { private Monitoring _monitoring; public Director(Monitoring monitoring) { _monitoring = monitoring; } public void Start(BaseJob baseJob) { baseJob.Accept(_monitoring); } } public class AutoMonitor: Monitoring { public override void Monitor(BaseJob baseJob) { Debug.WriteLine("发送前动作"); baseJob.Execute(); Debug.WriteLine("发送后动作"); } } public class SendEmail : BaseJob { public override void Accept(Monitoring monitoring) { monitoring.Monitor(this); } public override void Execute() { Debug.WriteLine("邮件发送"); } } public class SendMessage : BaseJob { public override void Accept(Monitoring monitoring) { monitoring.Monitor(this); } public override void Execute() { Debug.WriteLine("短信发送"); } } public class SendOther : BaseJob { public override void Accept(Monitoring monitoring) { monitoring.Monitor(this); } public override void Execute() { Debug.WriteLine("App发送"); } } public abstract class BaseJob { public abstract void Execute();public abstract void Accept(Monitoring monitoring); } public abstract class Monitoring { public abstract void Monitor(BaseJob baseJob); }
Monitoring monitoring = new AutoMonitor(); Director director = new Director(monitoring); director.Start(new SendEmail()); director.Start(new SendOther());
标签:BaseJob,Monitoring,void,模式,override,monitoring,public,访问者 From: https://www.cnblogs.com/morec/p/16882472.html