此篇文章主要是讲解如何手写SpringMVC的思路,其中会涉及到一部分的代码来辅助来说明思路,希望对想手写SpringMVC,而无从下手的朋友有些帮助。
一、先来一份SpringMVC的实现功能,然后通过代码来讲解我们手写SpringMVC需要写哪些东西
大多数的公司对代码的结构都是:
浏览器请求 ==》Controller层---》Service层---》Dao层 ==》数据库
数据请求到后,就会响应返回给浏览器展示。
本篇文章主要的目的是讲解SpringMVC,所以主要手写的部分是 Controller层---》Service层,考虑到大家对SpringMVC的使用有一定的了解,所以这部分只是罗列一部分的代码,来辅助说明需要哪些注解来解析这部分的代码。
Controller层:
@Controller
@RequestMapping("/busi")
public class BusiController{
@AutoWired
private BusiService busiService;
@RequestMapping("/query")
public void query(@RequestParam("name")String name) {
busiService.query(name);
}
}
Service层:
public class BusiService {
public String query(String name) ;
}
@Service
public class BusiServiceImpl {
public String query(String name) {
return "BusiServiceImpl " + name;
}
}
从上述的代码实现中发现代码中使用了下面的注解,同时也会在下面的列表会通过自定义形式来命令这些注解
@Controller @RequestMapping @Qualifier @RequestParam @Service | @CustomController @CustomRequestMapping @CustomQualifier @CustomRequestParam @CustomService |
其中没有实现@AutoWired的注解,使用了@Qualifier来讲解。如果大家对这部分的注解不理解,可以百度下。
二、手写SpringMVC对应的代码框架
现在有了上述的注解的对应,现在我们就先建立这些注解以及逻辑处理的代码结构。
这部分的代码可以实现在URL输入
http://localhost:8080/maven_handmvc/custom/query?name=chenyanwu&&age=21
返回的结果为:
三、大致实现思路
1、配置web.xml,加载我们自定义的DispatcherServlet
<?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/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5">
<display-name>maven_handmvc</display-name>
<servlet>
<servlet-name>DispatcherServlet</servlet-name>
<servlet-class>com.ygsoft.custom.servlet.DispatcherServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>DispatcherServlet</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
2、初始化阶段
在DispatcherServlet类中,实现下面几个步骤:
- 加载配置类
- 扫描当前项目下的所有文件
- 拿到扫描到的类,通过反射机制,实例化。并且放到ioc容器中(Map的键值对) beans
- 初始化path与方法的映射
@Override
public void init(ServletConfig config) throws ServletException {
System.out.println("init()............");
// 1.扫描需要的实例化的类
doScanPackage("com.ygsoft.custom");
System.out.println("当前文件下所有的class类.......");
for(String name: classNames) {
System.out.println(name);
}
// 2.实例化
doInstance();
System.out.println("当前实例化的对象信息.........");
for(Map.Entry<String, Object> map: beans.entrySet()) {
System.out.println("key:" + map.getKey() + "; value:" + map.getValue());
}
// 3.将IOC容器中的service对象设置给controller层定义的field上
doIoc();
// 4.建立path与method的映射关系
handlerMapping();
System.out.println("Controller层的path和方法映射.........");
for(Map.Entry<String, Object> map: handlerMap.entrySet()) {
System.out.println("key:" + map.getKey() + "; value:" + map.getValue());
}
}
3、运行阶段
每一次请求将会调用doGet或doPost方法,它会根据url请求去HandlerMapping中匹配到对应的Method,然后利用反射机制调用Controller中的url对应的方法,并得到结果返回。按顺序包括以下功能:
- 获取请求传入的参数并处理参数
- 通过初始化好的handlerMapping中拿出url对应的方法名,反射调用
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
System.out.println("doGet()............");
this.doPost(req, resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
System.out.println("doPost()............");
// 通过req获取请求的uri /maven_handmvc/custom/query
String uri = req.getRequestURI();
// /maven_handmvc
String context = req.getContextPath();
String path = uri.replaceAll(context, "");
// 通过当前的path获取handlerMap的方法名
Method method = (Method) handlerMap.get(path);
// 获取beans容器中的bean
MyController instance = (MyController) beans.get("/" + path.split("/")[1]);
// 处理参数
HandlerAdapterService ha = (HandlerAdapterService) beans.get("customHandlerAdapter");
Object[] args = ha.handle(req, resp, method, beans);
// 通过反射来实现方法的调用
try {
method.invoke(instance, args);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
如果大家有兴趣,可以按照这个思路来实现下,如何手写SpringMVC,其中可能描述不够清楚或者大家有疑问,可以留言,欢迎大家一起讨论学习。
扫描关注:全栈工程师成长记
一个可以交流的平台,目的是为了做一个影响最有影响力的人的平台。