手把手教你写一个SpringMVC框架
一、介绍
在日常的 web 开发中,熟悉 java 的同学一定知道,Spring MVC 可以说是目前最流行的框架,之所以如此的流行,原因很简单:编程简洁、上手简单!
我记得刚开始入行的时候,最先接触到的是Struts1 + Hibernate + Spring
来web系统的整体开发框架,简单的描述一下当时的编程心情:超难用,各种配置项很多,而且不容易快速入手!
之后,新的项目换成了Struts2 + hibernate + spring
来作为主体开发框架,Struts2
相比Struts1
编程要简单很多,而且加强了对拦截器与IoC
的支持,而在Struts1
中,这些特性是很难做的的!
然而随着Struts2
的使用量越来越广,业界爆出关于Struts2
的bug
和安全漏洞却越来越多!
文章图片
黑客们可以轻易的利用安全漏洞直接绕开安全防线,获取用的隐私数据,网名因个人信息泄露造成的经济损失高达 915 亿元!
至此很多开发者开始转到SpringMVC
框架阵营!
今天我们要介绍的主角就是SpringMVC
框架,刚开始玩这个的时候,给我最直接的感觉就是:很容易简单!
直接通过几个注解就可以完成方法的暴露,比起Struts2
中繁琐的xml
配置,SpringMVC
的使用可以说更加友好!
熟悉SpringMVC
框架的同学一定清楚下面这张图,
文章图片
【手把手教你写一个SpringMVC框架】这张图就是 SpringMVC 在处理 http 请求的整个流程中所做的一些事情。
- 1、用户发送请求至前端控制器DispatcherServlet
- 2、DispatcherServlet收到请求调用HandlerMapping处理器映射器。
- 3、处理器映射器根据请求url找到具体的处理器,生成处理器对象及处理器拦截器(如果有则生成)一并返回给DispatcherServlet。
- 4、DispatcherServlet通过HandlerAdapter处理器适配器调用处理器
- 5、执行处理器(Controller,也叫后端控制器)。
- 6、Controller执行完成返回ModelAndView
- 7、HandlerAdapter将controller执行结果ModelAndView返回给DispatcherServlet
- 8、DispatcherServlet将ModelAndView传给ViewReslover视图解析器
- 9、ViewReslover解析后返回具体View
- 10、DispatcherServlet对View进行渲染视图(即将模型数据填充至视图中)。
- 11、DispatcherServlet响应用户。
基于上面的流程,我们可以编写出一款简化版的
Spring MVC
框架,话不多说,直接撸起来!二、程序实践
首先上图!
文章图片
这个就是我们简易版的
Spring MVC
框架的实现流程图!- 1、首先创建一个
DispatcherServlet
类,在服务启动的时候,读取要扫描的包路径,然后通过反射将类信息存储到ioc
容器,同时通过@Autowired
注解,实现自动依赖注入,最后读取@RequestMapping
注解中的方法,将映射路径与类的关系存储到映射容器中。 - 2、当用户发起请求的时候,通过请求路径到
映射容器
中找到对应的执行类,然后调用具体的方法,发起逻辑处理,最后将处理结果返回给前端用户!
2.1、创建扫描注解 因为
Spring MVC
基本全部都是基于注解开发,因此我们事先也需要创建对应的注解,各个含义与Spring MVC
一致!- 控制层注解
/**
* 控制层注解
* @Controller
*/
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Controller {String value() default "";
}
- 请求路径注解
/**
* 请求路径注解
* @RequestMapping
*/
@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RequestMapping {String value() default "";
}
- 参数注解
/**
* 参数注解
* @RequestParam
*/
@Target({ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RequestParam {String value() default "";
}
- 服务层注解
/**
* 服务层注解
* @Controller
*/
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Service {String value() default "";
}
- 自动装载注解
/**
* 自动装载注解
* @Autowrited
*/
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Autowired {String value() default "";
}
2.2、编写 DispatcherServlet 类
DispatcherServlet
是一个Servlet
类,主要承担的任务是:接受前端用户的请求,然后进行转发,最后响应结果给前端用户!详细代码如下:
/**
* servlet跳转层
*/
@WebServlet(name = "DispatcherServlet",urlPatterns = "/*", loadOnStartup = 1, initParams = {@WebInitParam(name="scanPackage", value="https://www.it610.com/article/com.example.mvc")})
public class DispatcherServlet extends HttpServlet {private static final long serialVersionUID = 1L;
private static final Logger logger = LoggerFactory.getLogger(DispatcherServlet.class);
/**请求方法映射容器*/
private static List handlerMapping = new ArrayList<>();
/**
* 服务启动的时候,进行初始化,流程如下:
* 1、扫描指定包下所有的类
* 2、通过反射将类实例,放入ioc容器
* 3、通过Autowired注解,实现自动依赖注入,也就是set类中的属性
* 4、通过RequestMapping注解,获取需要映射的所有方法,然后将类信息存放到容器中
* @param config
* @throws ServletException
*/
@Override
public void init(ServletConfig config) throws ServletException {
try {
//1、扫描指定包下所有的类
String scanPackage = config.getInitParameter("scanPackage");
//1、扫描指定包下所有的类
List classNames = doScan(scanPackage);
//2、初始化所有类实例,放入ioc容器,也就是map对象中
Map iocMap = doInstance(classNames);
//3、实现自动依赖注入
doAutowired(iocMap);
//5、初始化方法mapping
initHandleMapping(iocMap);
} catch (Exception e) {
logger.error("dispatcher-servlet类初始化失败!",e);
throw new ServletException(e.getMessage());
}
}/**
* @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
*/
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {
doPost(request, response);
}/**
* @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
*/
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws IOException {
//跳转
doDispatch(request, response);
}/**
* 扫描指定包下的类文件
* @param packageName
* @return
*/
private List doScan(String packageName){
if(StringUtils.isBlank(packageName)){
throw new RuntimeException("mvc配置文件中指定扫描包名为空!");
}
return PackageHelper.getClassName(packageName);
}private Map doInstance(List classNames) {
Map iocMap = new HashMap<>();
if(!CollectionUtils.isNotEmpty(classNames)){
throw new RuntimeException("获取的类为空!");
}
for (String className : classNames) {
try {
//通过反射机制构造对象
Class> clazz = Class.forName(className);
if(clazz.isAnnotationPresent(Controller.class)){
//将类名第一个字母小写
String baneName = firstLowerCase(clazz.getSimpleName());
iocMap.put(baneName, clazz.newInstance());
}else if(clazz.isAnnotationPresent(Service.class)){
//服务层注解判断
Service service = clazz.getAnnotation(Service.class);
String beanName = service.value();
//如果该注解上没有自定义类名,则默认首字母小写
if(StringUtils.isBlank(beanName)){
beanName = clazz.getName();
}
Object instance = clazz.newInstance();
iocMap.put(beanName, instance);
//如果注入的是接口,可以巧妙的用接口的类型作为key
Class>[] interfaces = clazz.getInterfaces();
for (Class> clazzInterface : interfaces) {
iocMap.put(clazzInterface.getName(), instance);
}
}
} catch (Exception e) {
logger.error("初始化mvc-ioc容器失败!",e);
throw new RuntimeException("初始化mvc-ioc容器失败!");
}
}
return iocMap;
}/**
* 实现自动依赖注入
* @throws Exception
*/
private void doAutowired(Map iocMap) {
if(!MapUtils.isNotEmpty(iocMap)){
throw new RuntimeException("初始化实现自动依赖失败,ioc为空!");
}
for(Map.Entry entry : iocMap.entrySet()){
//获取对象下所有的属性
Field[] fields = entry.getValue().getClass().getDeclaredFields();
for (Field field : fields) {
//判断字段上有没有@Autowried注解,有的话才注入
if(field.isAnnotationPresent(Autowired.class)){
try {
Autowired autowired = field.getAnnotation(Autowired.class);
//获取注解上有没有自定义值
String beanName = autowired.value().trim();
if(StringUtils.isBlank(beanName)){
beanName = field.getType().getName();
}
//如果想要访问到私有的属性,我们要强制授权
field.setAccessible(true);
field.set(entry.getValue(), iocMap.get(beanName));
} catch (Exception e) {
logger.error("初始化实现自动依赖注入失败!",e);
throw new RuntimeException("初始化实现自动依赖注入失败");
}
}
}
}
}/**
* 初始化方法mapping
*/
private void initHandleMapping(Map iocMap){
if(!MapUtils.isNotEmpty(iocMap)){
throw new RuntimeException("初始化实现自动依赖失败,ioc为空");
}
for(Map.Entry entry:iocMap.entrySet()){
Class> clazz = entry.getValue().getClass();
//判断是否是controller层
if(!clazz.isAnnotationPresent(Controller.class)){
continue;
}
String baseUrl = null;
//判断类有没有requestMapping注解
if(clazz.isAnnotationPresent(RequestMapping.class)){
RequestMapping requestMapping = clazz.getAnnotation(RequestMapping.class);
baseUrl= requestMapping.value();
}
Method[] methods = clazz.getMethods();
for (Method method : methods) {
//判断方法上有没有requestMapping
if(!method.isAnnotationPresent(RequestMapping.class)){
continue;
}
RequestMapping requestMethodMapping = method.getAnnotation(RequestMapping.class);
//"/+",表示将多个"/"转换成"/"
String regex = (baseUrl + requestMethodMapping.value()).replaceAll("/+", "/");
Pattern pattern = Pattern.compile(regex);
handlerMapping.add(new RequestHandler(pattern, entry.getValue(), method));
}
}
}/**
* servlet请求跳转
* @param request
* @param response
* @throws IOException
*/
private void doDispatch(HttpServletRequest request, HttpServletResponse response) throws IOException {
try {
request.setCharacterEncoding("UTF-8");
response.setHeader("Cache-Control", "no-cache");
response.setHeader("Pragma", "no-cache");
response.setDateHeader("Expires", -1);
response.setContentType("text/html");
response.setHeader("content-type", "text/html;
charset=UTF-8");
response.setCharacterEncoding("UTF-8");
RequestHandler handle = getHandleMapping(request);
if(Objects.isNull(handle)){
//异常请求地址
logger.warn("异常请求地址!地址:" + request.getRequestURI());
response.getWriter().append("error request url");
return;
}
//获取参数列表
Object[] paramValues = RequestParamHelper.buildRequestParam(handle, request, response);
Object result = handle.getMethod().invoke(handle.getController(), paramValues);
if(result != null){
PrintWriter out = response.getWriter();
out.println(result);
out.flush();
out.close();
}
} catch (Exception e) {
logger.error("接口请求失败!",e);
PrintWriter out = response.getWriter();
out.println("请求异常,请稍后再试");
out.flush();
out.close();
}
}/**
* 将类名第一个字母小写
* @param clazzName
* @return
*/
private String firstLowerCase(String clazzName){
char[] chars = clazzName.toCharArray();
chars[0] += 32;
return String.valueOf(chars);
}/**
* 获取用户请求方法名
* 与handlerMapping中的路径名进行匹配
* @param request
* @return
*/
private RequestHandler getHandleMapping(HttpServletRequest request){
if(CollectionUtils.isNotEmpty(handlerMapping)){
//获取用户请求路径
String url = request.getRequestURI();
String contextPath = request.getContextPath();
String serviceUrl = url.replace(contextPath, "").replaceAll("/+", "/");
for (RequestHandler handle : handlerMapping) {
//正则匹配请求方法名
Matcher matcher = handle.getPattern().matcher(serviceUrl);
if(matcher.matches()){
return handle;
}
}
}
return null;
}
}
这里要重点介绍一下初始化阶段所做的操作!
DispatcherServlet
在服务启动阶段,会调用init
方法进行服务初始化,此阶段所做的事情主要有以下内容:- 1、扫描指定包下所有的类信息,返回的结果主要是
包名 + 类名
- 2、通过反射机制,将类进行实例化,将类实例化对象存储到
ioc
容器中,其中key
是类名(小些驼峰),value
是类对象 - 3、通过
Autowired
注解找到类对象中的属性,通过小驼峰从ioc
容器中寻找对应的属性值,然后进行set
操作 - 4、通过
Controller
和RequestMapping
注解寻找需要暴露的方法,并获取对应的映射路径,最后将映射路径 - 5、最后,当前端用户发起一个请求时,
DispatcherServlet
获取到请求路径之后,通过与RequestMapping
中的路径进行匹配,找到对应的controller
类中的方法,然后通过invoke
完成方法调用,将调用结果返回给前端!
DispatcherServlet
编写完成之后,紧接着我们需要编写对应的controller
控制类来接受前端用户请求,下面我们以用户登录为例,程序示例如下:- 编写一个
LoginController
控制类,接受前端用户调用
@Controller
@RequestMapping("/user")
public class LoginController {@Autowired
private UserService userService;
/**
* 用户登录
* @param request
* @param response
* @param userName
* @param userPwd
* @return
*/
@RequestMapping("/login")
public String login(HttpServletRequest request, HttpServletResponse response,
@RequestParam("userName") String userName,
@RequestParam("userPwd") String userPwd){
boolean result = userService.login(userName, userPwd);
if(result){
return "登录成功!";
} else {
return "登录失败!";
}
}
}
- 编写一个
UserService
服务类,用于判断账户、密码是否正确
public interface UserService {/**
* 登录
* @param userName
* @param userPwd
* @return
*/
boolean login(String userName, String userPwd);
}
@Service
public class UserServiceImpl implements UserService {@Override
public boolean login(String userName, String userPwd) {
if("zhangsan".equals(userName) && "123456".equals(userPwd)){
return true;
} else {
return false;
}
}
}
最后,将项目打包成
war
,通过tomcat
启动服务!在浏览器中访问
http://localhost:8080/user/login?userName=hello&userPwd=123
,结果显示如下:文章图片
当我们将
userName
和userPwd
换成正确的数据,访问地址如下:http://localhost:8080/user/login?userName=zhangsan&userPwd=123456
文章图片
可以很清晰的看到,服务调用正常!
三、总结
本文主要以
Spring MVC
框架为背景,手写了一个简易版的Spring MVC
框架,虽然功能简陋了一点,但是基本无张俱全,里面讲解了ioc
和自动依赖注入的实现过程,还有前端发起一个路径请求,是如何映射到对应的controller
类中的方法上!当然实际的
Spring MVC
框架的跳转流程比这个复杂很多很多,里面包括各种拦截器、权限安全管理等等,在后面的文章,小编也会陆续进行详细介绍!下面是手写的简易版
Spring MVC
框架源码地址,感兴趣的朋友,关注下方公众号,并回复【cccc8】即可获取!推荐阅读
- 手写简版kedis分布式key及value服务的实现及配置
- 求职特训营火热来袭|求职特训营火热来袭 阿里大咖教你制作专业简历!
- 一篇文章教你读懂量子通信,真的很有趣
- 通过|通过 AssemblyScript 编写 wasm
- 手写|手写 Vue2 系列 之 patch —— diff
- 纯后端如何写前端(我用了低代码平台)
- 技术交流|从硬件到软件,教你从零搭建智慧农业大脑
- hive|hive3.0惊天大bug发现,grouping()函数只能小写,大写直接报错,hive不是不区分大小写吗.
- 伪经验;不称职面试官的面试经
- 教你bat脚本一键配置java开发环境