在我们的日常工作中,经常会用到Spring、Spring Boot、Spring Cloud、Struts、Mybatis、Hibernate等开源框架,有了这些框架的诞生,平时的开发工作量也是变得越来越轻松,我们用 Spring Boot
分分钟可以新建一个Web项目。
记得自己刚开始工作的时候还是在用Servlet
写Web
项目,自己写数据库连接池,用原生JDBC
操作数据库,好了不发散了。回到这篇文章的主题,今天通过手写Spring框架,帮大家深入了解一下Spring的工作机制,文中涉及的代码只用来帮助大家理解Spring,不会在线上使用,有不严谨的地方还请大家掠过。
项目结构
框架部分实现
- 为了区分框架部分代码和业务部分代码,我们将这两部分分别划分在不同的包内
com.mars.demo
和com.mars.framework
,以便随后只扫描业务代码。 - 这里是自己手写Spring框架,所以不会引入任何Spring项目相关的包。
- 由于是一个Web项目,所有我们需要引入
servlet-api
包,仅供编译器使用,所有配置scope
为provided
。
新建一个Servlet
首先新建一个 HttpServlet 的实现类 MarsDispatcherServlet
,用来接收请求。
public class MarsDispatcherServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
this.doPost(req, resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//6. 处理请求
}
@Override
public void init(ServletConfig config) throws ServletException {
}
配置web.xml
<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
"http://java.sun.com/dtd/web-app_2_3.dtd" >
<web-app>
<display-name>Spring Mvc Education</display-name>
<servlet>
<servlet-name>marsmvc</servlet-name>
<servlet-class>com.mars.framework.servlet.MarsDispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>application.properties</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>marsmvc</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
</web-app>
- 首先配置了一个 servlet, 名字是 marsmvc, 类全路径是
com.mars.framework.servlet.MarsDispatcherServlet
。 - 设置了初始化参数名和值(这里的值是整个项目的配置文件)。
- 配置
load-on-startup
, 标记容器是否在启动的时候就加载这个servlet(实例化并调用其init()方法)。 - 配置
servlet-mapping
, 将所有请求转发到这个servlet处理。
配置application.properties
scanPackage=com.mars.demo
这个比较好理解,仅配置了一项内容,意思是要扫描的包,随后我们会获取这个值去加载容器。
定义我们常用的注解
- MarsAutowired
- MarsController
- MarsRequestMapping
- MarsRequestParam
- MarsService
这里仅列举两个,其他都大同小异,需要源码的可以去我的代码仓库fork。
@Target({
ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MarsController {
String value() default "";
}
@Target({
ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MarsRequestMapping {
String value() default "";
}
充实Servlet功能
先列出框架在初始化的时候都要做那些事情
- 加载配置文件
- 扫描所有相关联的类
- 初始化所有相关联的类,并且将其保存在IOC容器里面
- 执行依赖注入(把加了@Autowired注解的字段赋值)
- 构造HandlerMapping,将URL和Method进行关联
接下来我们一步步完成上面的操作
@Override
public void init(ServletConfig config) throws ServletException {
System.out.println("===================")