自己实现SpringMVC 底层机制[四]

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 自己实现SpringMVC 底层机制[四]

自己实现SpringMVC 底层机制[四]


实现任务阶段7- 完成简单视图解析


功能说明:通过方法返回的String, 转发或者重定向到指定页面。


完成任务说明


用户输入白骨精,可以登录成功, 否则失败。

根据登录的结果, 可以重定向或者请求转发到login_ok.jsp / login_error.jsp, 并显示妖怪名.


测试页面

3cd29a6c780e4d5abbc5bcf6daae19f5.png

aa81f4e3c793407a8b2f345485c8e727.png

a4a454da4c394e50add59a12b5da678c.png


代码实现


修改my-springmvc\src\main\java\com\service\MonsterService.java

public interface MonsterService {
    public List<Monster> listMonsters();
    public List<Monster> findMonstersByName(String name);
    public boolean login(String name);
}


修改my-springmvc\src\main\java\com\service\impl\MonsterServiceImpl.java

@Service
public class MonsterServiceImpl implements MonsterService {
    @Override
    public boolean login(String name) {
           if ("白骨精".equals(name)) {
                    return true;
            } else {
                 return false;
            }
    }
}


修改my-springmvc\src\main\java\com\controller\MonsterController.java ,增加方法.

处理妖怪登录的方法,返回要请求转发/重定向的字符串

    @RequestMapping("/monster/login")
    public String login(HttpServletRequest request,HttpServletResponse response,  String mName) {
        System.out.println("--接收到mName---" + mName);
        //将mName设置到request域
        request.setAttribute("mName", mName);
        boolean b = monsterService.login(mName);
        if (b) {//登录成功!
            //测试重定向
            //return "redirect:/login_ok.jsp";
            //测试默认的方式-forward
            return "forward:/login_ok.jsp";
        } else {//登录失败
            return "forward:/login_error.jsp";
        }
    }


增加my-springmvc\src\main\webapp\login_ok.jsp

创建登录成功页面返回登录用户信息。

<%@ page contentType="text/html;charset=UTF-8" language="java" isELIgnored="false" %>
<html>
    <head>
        <title>登录成功</title>
    </head>
    <body>
        <h1>登录成功</h1>
        欢迎你: ${requestScope.mName}
        <!--从request域中动态获取登录用户信息-->
    </body>
</html>


创建my-springmvc\src\main\webapp\login_error.jsp

创建创建登录失败页面返回登录用户信息。

<%@ page contentType="text/html;charset=UTF-8" language="java" isELIgnored="false" %>
<html>
    <head>
        <title>登录失败</title>
    </head>
    <body>
        <h1>登录失败</h1>
        sorry, 登录失败 ${requestScope.mName}
          <!--从request域中动态获取登录用户信息-->
    </body>
</html>


修改my-springmvc\src\main\java\com\myspringmvc\servlet\MyDispatcherServlet.java

编写方法,完成分发请求任务。提交可能出现中文格式,为防止出现乱码我们需要设置编码格式。

    private void executeDispatch(HttpServletRequest request,  HttpServletResponse response) {
        MyHandler myHandler = getMyHandler(request);
        try {
            if (null == myHandler) {//说明用户请求的路径/资源不存在
                response.getWriter().print("<h1>404 NOT FOUND</h1>");
            } else {//匹配成功, 反射调用控制器的方法
                //目标将: HttpServletRequest 和 HttpServletResponse封装到参数数组
                //1. 得到目标方法的所有形参参数信息[对应的数组]
                Class<?>[] parameterTypes =   myHandler.getMethod().getParameterTypes();
                //2. 创建一个参数数组[对应实参数组], 在后面反射调用目标方法时,会使用到
                Object[] params =  new Object[parameterTypes.length];
                //3遍历parameterTypes形参数组,根据形参数组信息,将实参填充到实参数组
                for (int i = 0; i < parameterTypes.length; i++) {
                    //取出每一个形参类型
                    Class<?> parameterType = parameterTypes[i];
                    //如果这个形参是HttpServletRequest, 将request填充到params
                    //在原生SpringMVC中,是按照类型来进行匹配,这里简化使用名字来进行匹配
                    if ("HttpServletRequest".equals(parameterType.getSimpleName())) {
                        params[i] = request;
                    } else if ("HttpServletResponse".equals(parameterType.getSimpleName())) {
                        params[i] = response;
                    }
                }
                //将http请求参数封装到params数组中, 要注意填充实参的时候,顺序问题
                //1. 获取http请求的参数集合
                //http://localhost:8080/monster/find?name=牛魔王&hobby=打篮球&hobby=喝酒
                //2. 返回的Map<String,String[]> String:表示http请求的参数名
                //   String[]:表示http请求的参数值是数组
                //处理提交的数据中文乱码
                request.setCharacterEncoding("utf-8");
                Map<String, String[]> parameterMap =   request.getParameterMap();
                //2. 遍历parameterMap 将请求参数,按照顺序填充到实参数组params
                for (Map.Entry<String, String[]> entry : parameterMap.entrySet()) {
                    //取出key,这name就是对应请求的参数名
                    String name = entry.getKey();
                    //说明:这里只考虑提交的参数是单值的情况,即不考虑类似checkbox提示的数据
                    String value = entry.getValue()[0];
                    //我们得到请求的参数对应目标方法的第几个形参,然后将其填充
                    //这里专门编写一个方法,得到请求的参数对应的是第几个形参
                    int indexRequestParameterIndex =  getIndexRequestParameterIndex(myHandler.getMethod(), name);
                    if (indexRequestParameterIndex != -1) {//找到对应的位置
                        params[indexRequestParameterIndex] = value;
                    } else {//说明并没有找到@RequestParam注解对应的参数,就会使用默认的机制进行配置[待..]
                        //思路
                        //1. 得到目标方法的所有形参的名称-专门编写方法获取形参名
                        //2. 对得到目标方法的所有形参名进行遍历,如果匹配就把当前请求的参数值,填充到params
                        List<String> parameterNames =
                                getParameterNames(myHandler.getMethod());
                        for (int i = 0; i < parameterNames.size(); i++) {
                            //如果请求参数名和目标方法的形参名一样,说明匹配成功
                            if (name.equals(parameterNames.get(i))) {
                                params[i] = value;//填充到实参数组
                                break;
                            }
                        }
                    }
                }
                /**
                 * 解读
                 * 1. 下面这样写法,其实是针对目标方法是 m(HttpServletRequest request , HttpServletResponse response)
                 * 2. 这里准备将需要传递给目标方法的 实参=>封装到参数数组=》然后以反射调用的方式传递给目标方法
                 * 3. public Object invoke(Object obj, Object... args)..
                 */
                //myHandler.getMethod() .invoke(myHandler.getController(),request,response);
                //反射调用目标方法
                Object result = myHandler.getMethod().invoke(myHandler.getController(), params);
                //这里就是对返回的结果进行解析=>原生springmvc 可以通过视图解析器来完成
                if (result instanceof String) {
                    String viewName = (String) result;
                    if(viewName.contains(":")){//说明你返回的String 结果forward:/login_ok.jsp 或者 redirect:/xxx/xx/xx.xx
                        String viewType = viewName.split(":")[0];//forward | redirect
                        String viewPage = viewName.split(":")[1];//是你要跳转的页面名
                        //判断是forward 还是 redirect
                        if("forward".equals(viewType)) {//说明你希望请求转发
                            request.getRequestDispatcher(viewPage) .forward(request,response);
                        } else if("redirect".equals(viewType)) {//说明你希望重定向
                            response.sendRedirect(viewPage);
                        }
                    } else {//默认是请求转发
                        request.getRequestDispatcher(viewName) .forward(request,response);
                    }
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


完成测试


启动tomcat,浏览器输入http://localhost:8080/monster/login?mName=白骨精

f198fbdb91364e049a14e73f4d9ce752.png


实现任务阶段8- 完成返回JSON 格式数据-@ResponseBody


功能说明:通自定义@ResponseBody 返回JSON 格式数据。


代码实现


创建my-springmvc\src\main\java\com\myspringmvc\annotation\ResponseBody.java

@Target(ElementType.METHOD)//该注解只能声明在一个类的方法前
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface ResponseBody {
}


创建my-springmvc\src\main\java\com\controller\MonsterController.java, 增加方法

    @RequestMapping("/monster/list/json")
    @ResponseBody
    public List<Monster> listMonsterByJson(HttpServletRequest request,  HttpServletResponse response) {
        List<Monster> monsters = monsterService.listMonster();
        return monsters;
    }


修改my-springmvc\pom.xml

要用json格式数据,我们可以用springMVC自己的,前提是我们需要引入jackson 使用他的工具类可以进行json操作。

  <!--引入jackson 使用他的工具类可以进行json操作 -->
<dependency>
       <groupId>com.fasterxml.jackson.core</groupId>
       <artifactId>jackson-databind</artifactId>
       <version>2.12.4</version>
</dependency>


修改my-springmvc\src\main\java\com\myspringmvc\servlet\MyDispatcherServlet.java

编写方法,完成分发请求任务

    private void executeDispatch(HttpServletRequest request,  HttpServletResponse response) {
        MyHandler myHandler = getMyHandler(request);
        try {
            if (null == myHandler) {//说明用户请求的路径/资源不存在
                response.getWriter().print("<h1>404 NOT FOUND</h1>");
            } else {//匹配成功, 反射调用控制器的方法
                //目标将: HttpServletRequest 和 HttpServletResponse封装到参数数组
                //1. 得到目标方法的所有形参参数信息[对应的数组]
                Class<?>[] parameterTypes =   myHandler.getMethod().getParameterTypes();
                //2. 创建一个参数数组[对应实参数组], 在后面反射调用目标方法时,会使用到
                Object[] params =  new Object[parameterTypes.length];
                //3遍历parameterTypes形参数组,根据形参数组信息,将实参填充到实参数组
                for (int i = 0; i < parameterTypes.length; i++) {
                    //取出每一个形参类型
                    Class<?> parameterType = parameterTypes[i];
                    //如果这个形参是HttpServletRequest, 将request填充到params
                    //在原生SpringMVC中,是按照类型来进行匹配,这里简化使用名字来进行匹配
                    if ("HttpServletRequest".equals(parameterType.getSimpleName())) {
                        params[i] = request;
                    } else if ("HttpServletResponse".equals(parameterType.getSimpleName())) {
                        params[i] = response;
                    }
                }
                //将http请求参数封装到params数组中, 要注意填充实参的时候,顺序问题
                //1. 获取http请求的参数集合
                //http://localhost:8080/monster/find?name=牛魔王&hobby=打篮球&hobby=喝酒
                //2. 返回的Map<String,String[]> String:表示http请求的参数名
                //   String[]:表示http请求的参数值是数组
                //处理提交的数据中文乱码
                request.setCharacterEncoding("utf-8");
                Map<String, String[]> parameterMap =   request.getParameterMap();
                //2. 遍历parameterMap 将请求参数,按照顺序填充到实参数组params
                for (Map.Entry<String, String[]> entry : parameterMap.entrySet()) {
                    //取出key,这name就是对应请求的参数名
                    String name = entry.getKey();
                    //说明:这里只考虑提交的参数是单值的情况,即不考虑类似checkbox提示的数据
                    String value = entry.getValue()[0];
                    //我们得到请求的参数对应目标方法的第几个形参,然后将其填充
                    //这里专门编写一个方法,得到请求的参数对应的是第几个形参
                    int indexRequestParameterIndex =  getIndexRequestParameterIndex(myHandler.getMethod(), name);
                    if (indexRequestParameterIndex != -1) {//找到对应的位置
                        params[indexRequestParameterIndex] = value;
                    } else {//说明并没有找到@RequestParam注解对应的参数,就会使用默认的机制进行配置
                        //思路
                        //1. 得到目标方法的所有形参的名称-专门编写方法获取形参名
                        //2. 对得到目标方法的所有形参名进行遍历,如果匹配就把当前请求的参数值,填充到params
                        List<String> parameterNames =  getParameterNames(myHandler.getMethod());
                        for (int i = 0; i < parameterNames.size(); i++) {
                            //如果请求参数名和目标方法的形参名一样,说明匹配成功
                            if (name.equals(parameterNames.get(i))) {
                                params[i] = value;//填充到实参数组
                                break;
                            }
                        }
                    }
                }
                /**
                 * 解读
                 * 1. 下面这样写法,其实是针对目标方法是 m(HttpServletRequest request , HttpServletResponse response)
                 * 2. 这里准备将需要传递给目标方法的 实参=>封装到参数数组=》然后以反射调用的方式传递给目标方法
                 * 3. public Object invoke(Object obj, Object... args)..
                 */
                //myHandler.getMethod() .invoke(myHandler.getController(),request,response);
                //反射调用目标方法
                Object result = myHandler.getMethod().invoke(myHandler.getController(), params);
                //这里就是对返回的结果进行解析=>原生springmvc 可以通过视图解析器来完成
                if (result instanceof String) {
                    String viewName = (String) result;
                    if(viewName.contains(":")){//说明你返回的String 结果forward:/login_ok.jsp 或者 redirect:/xxx/xx/xx.xx
                        String viewType = viewName.split(":")[0];//forward | redirect
                        String viewPage = viewName.split(":")[1];//是你要跳转的页面名
                        //判断是forward 还是 redirect
                        if("forward".equals(viewType)) {//说明你希望请求转发
                            request.getRequestDispatcher(viewPage)
                                    .forward(request,response);
                        } else if("redirect".equals(viewType)) {//说明你希望重定向
                            response.sendRedirect(viewPage);
                        }
                    } else {//默认是请求转发
                        request.getRequestDispatcher(viewName)
                                .forward(request,response);
                    }
                }//这里还可以扩展
                else if(result instanceof ArrayList) {//如果是ArrayList
                    //判断目标方法是否有@ResponseBody
                    Method method = myHandler.getMethod();
                    if(method.isAnnotationPresent(ResponseBody.class)) {
                        //把result [ArrayList] 转成json格式数据-》返回
                        //这里我们需要使用jackson包下的工具类可以轻松的搞定.
                        ObjectMapper objectMapper = new ObjectMapper();
                        String resultJson =
                                objectMapper.writeValueAsString(result);
                        response.setContentType("text/html;charset=utf-8");
                        //这里简单的处理,就直接返回
                        PrintWriter writer = response.getWriter();
                        writer.write(resultJson);
                        writer.flush();
                        writer.close();
                    }
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


完成测试


启动Tomcat,浏览器输入http://localhost:8080/monster/list/json

image.png

也可以使用postman进行测试。


小结


感兴趣的也可以自己进行debug源码解析,看看springMVC的执行流程,印象会更加深刻。不用每一步都debug查看,主要看主要执行部分就可以,看的太多了反而会混乱,可以按照下面的springMVC底层解析图进行debug。谢谢。

5b55b83714c7469f886c89f0ea38a383.png

相关文章
|
12月前
|
缓存 Java 数据库
Spring框架(一) 底层核心原理解析
这个才是我们想要看的结果 ,我们可以简单分析一下 , userServiceBase的test1()方法也是有事务存在的 , 同时userServiceBase也是一个Bean , 它最终也会产生一个代理对象去当做一个Bean , 碎玉UserService而言 , 我要给userServiceBase这个属性去赋值 , 那么他肯定要从Spring容器中找到一个userServiceBase的一个Bean来赋值 , 所以他找到的就是Spring事务所产生的userServiceBase的代理对象 , 所以这个注解就是有用的
138 0
|
5月前
|
安全 Java 程序员
Spring框架的核心特性是什么?
【4月更文挑战第30天】Spring 的特性
223 0
|
2月前
|
Java 开发者 Spring
"揭秘SpringBoot魔法SPI机制:一键解锁服务扩展新姿势,让你的应用灵活飞天!"
【8月更文挑战第11天】SPI(Service Provider Interface)是Java的服务提供发现机制,用于运行时动态查找和加载服务实现。SpringBoot在其基础上进行了封装和优化,通过`spring.factories`文件提供更集中的配置方式,便于框架扩展和组件替换。本文通过定义接口`HelloService`及其实现类`HelloServiceImpl`,并在`spring.factories`中配置,结合`SpringFactoriesLoader`加载服务,展示了SpringBoot SPI机制的工作流程和优势。
42 5
|
5月前
|
机器学习/深度学习 运维 Java
江帅帅:Spring Boot 底层级探索系列 02 - 自动配置的底层逻辑
江帅帅:Spring Boot 底层级探索系列 02 - 自动配置的底层逻辑
42 0
|
11月前
|
Dubbo Java 应用服务中间件
阿里一面:说一说Java、Spring、Dubbo三者SPI机制的原理和区别
大家好,我是三友~~ 今天来跟大家聊一聊Java、Spring、Dubbo三者SPI机制的原理和区别。 其实我之前写过一篇类似的文章,但是这篇文章主要是剖析dubbo的SPI机制的源码,中间只是简单地介绍了一下Java、Spring的SPI机制,并没有进行深入,所以本篇就来深入聊一聊这三者的原理和区别。
|
12月前
|
存储 Java 数据库连接
【Spring传播机制底层原理】
【Spring传播机制底层原理】
|
XML Java 应用服务中间件
自己实现SpringMVC 底层机制[一]
自己实现SpringMVC 底层机制[一]
77 1
|
12月前
|
前端开发 Java 程序员
SpringMVC的工作原理及底层剖析,你值得一看
剩下的都在刚开始那段代码中了,其实这个也没啥就是简单的看看MVC工作的时候底层在干啥,不合适的地方多多指教。
52 0
|
Java 应用服务中间件 容器
自己实现SpringMVC 底层机制[二]
自己实现SpringMVC 底层机制[二]
60 0
|
前端开发 Java 应用服务中间件
自己实现SpringMVC 底层机制[三]
自己实现SpringMVC 底层机制[三]
41 0