1. 定义
委派模式的基本作用就是负责任务调度和分配任务,跟代理模式很像,可以看做是一种特殊情况下的静态代理的全权,但是代理模式注重过程,而委派模式注重结果
委派模式并不是23种设计模式
2. 适用场景
基本作用就是负责任务的调用和分配任务,跟代理模式很像,可以看做是一种特殊情况下的静态代理的全权代理,但是代理模式注重过程,而委派模式注重结果
委派模式在 Spring 中应用非常多,大家常用的 DispatcherServlet 其实就是用到了委派模式
现实生活中也常有委派的场景发生,例如:老板(Boss)给项目经理(Leader)下达任务,项目经理会根据实际情况给每个员工派发工作任务,待员工把工作任务完成之后,再由项目经理汇报工作进度和结果给老板
3. 代码实现
- IEmployee 员工接口
public interface IEmployee {
/**
* 工作
*/
void doing(String command);
}
- 员工 EmployeeA 类
public class EmployeeA implements IEmployee {
@Override
public void doing(String command) {
System.out.println("我擅长项目安全测试, 开始对项目进行" + command);
}
}
- 员工 EmployeeB 类
public class EmployeeB implements IEmployee {
@Override
public void doing(String command) {
System.out.println("我擅长架构设计, 开始进行项目" + command + "工作");
}
}
- 项目经理 Leader 类
public class Leader implements IEmployee {
private Map<String, IEmployee> register= new HashMap<>();
public Leader() {
register.put("安全测试", new EmployeeA());
register.put("架构设计", new EmployeeB());
}
public void doing(String command) {
register.get(command).doing(command);
}
}
- Boss 类下达命令
public class Boss {
public static void main(String[] args) {
Leader leader = new Leader();
leader.doing("安全测试");
leader.doing("架构设计");
}
}
4. SpringMVC 中的 Dispacher
有了 Dispatcher 做委派分发,用户在浏览器输入的 url 回调分发到对应的控制器中
下面我们再来还原一下 SpringMVC 的 DispatcherServlet 是如何实现委派模式的
- 业务类 MemberController
public class MemberController {
public void getMemberById(String mid){
}
}
- OrderController 类
public class OrderController {
public void getOrderById(String mid){
}
}
- SystemController 类
public class SystemController {
public void logout() {
}
}
- Dispacher 类
public class DispatcherServlet extends HttpServlet {
private void doDispatch(HttpServletRequest request, HttpServletResponse response) throws
Exception {
String uri = request.getRequestURI();
String mid = request.getParameter("mid");
if ("getMemberById".equals(uri)) {
new MemberController().getMemberById(mid);
} else if ("getOrderById".equals(uri)) {
new OrderController().getOrderById(mid);
} else if ("logout".equals(uri)) {
new SystemController().logout();
} else {
response.getWriter().write("404 Not Found!!");
}
}
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
try {
doDispatch(req, resp);
} catch (Exception e) {
e.printStackTrace();
}
}
}
- 配置 web.xml 文件
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://java.sun.com/xml/ns/j2ee"
xmlns:javaee="http://java.sun.com/xml/ns/javaee"
xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd" version="2.4">
<display-name>My Web Application</display-name>
<servlet>
<servlet-name>delegateServlet</servlet-name>
<servlet-class>com.zhunongyun.toalibaba.designpatterns.delegate.mvc.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>delegateServlet</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
</web-app>
一个完整的委派模式就实现出来了.当然在 Spring 中运用到委派模式不仅于此,还有很多.小伙伴们可以通过命名就可以识别,在 Spring 源码中,只要以 Delegate 结尾的都是实现了委派模式.例如:BeanDefinitionParserDelegate 根据不同类型委派不同的逻辑解析 BeanDefinition