springMvc框架实现

一 springMvc核心功能

springMvc是基于分层MVC设计模式的spring实现,他的核心功能是:

1 url跳转

2 ioc

3 aop(本次暂不实现)

 

二springMvc工作流程图

 

三 接下来,让我们写代码实现springMvc框架

通过注解方式实现了跳转和ioc两个核心功能,工程结构如下:

分别创建注解@Controller

package com.hongsen.annotation;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Controller {
    String value();
}

 创建@RequestMapping注解

package com.hongsen.annotation;

@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RequestMapping {
    String value();
}

 创建@resource注解

package com.hongsen.annotation;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Resource {
    String value();
}

 创建@Service注解

package com.hongsen.annotation;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Service {
    String value();
}

 

创建servlet,代码如下:

package com.hongsen.servlet;

/**
 * springMvc 作用
 * <p>
 * 1 跳转
 * 2 ioc
 * 3 aop
 *
 * @author wanghongsen 2018.07.27
 */

public class DispatcherServlet extends HttpServlet {

    private static final long serialVersionUID = 1L;
    private List<String> classList = new ArrayList<>();
    private Map<String, Object> beanInstanceMap = new HashMap<>();
    private Map<String, Method> urlHandlerMap = new HashMap<>();

    /**
     * servlet初始化方法
     * 1 scan扫描包路径
     * 2 构造所有类集合 (list<String>)
     * 3 构造所有bean实例集合 (beanName-> bean)
     * 4 构造所有请求handler跳转(uri -> method)
     */
    public void init() throws ServletException {
        System.out.println("init() start");
        String scanBasePath = "com.hongsen";
        scan(scanBasePath);
        initBeanInstance();
        ioc();
        initUrlHandlerMap();
    }

    private void scan(String scanBasePath) {
        URL url = this.getClass().getResource("/" + scanBasePath.replace(".", "/"));
        String baseFilePath = url.getFile();
        File baseFile = new File(baseFilePath);
        String[] subFilePaths = baseFile.list();
        for (String subFilePath : subFilePaths != null ? subFilePaths : new String[0]) {
            File subFile = new File(baseFilePath + subFilePath);
            if (subFile.isDirectory()) {
                scan(scanBasePath + "." + subFilePath);
            } else {
                classList.add(scanBasePath + "." + subFilePath);
            }
        }
    }
private void initBeanInstance() { for (String beanName : classList) { try { beanName = beanName.replace(".class", ""); Class beanClass = Class.forName(beanName); try { if (beanClass.isAnnotationPresent(Controller.class)) { Controller controller = (Controller) beanClass.getAnnotation(Controller.class); beanInstanceMap.put(controller.value(), beanClass.newInstance()); } else if (beanClass.isAnnotationPresent(Service.class)) { Service service = (Service) beanClass.getAnnotation(Service.class); beanInstanceMap.put(service.value(), beanClass.newInstance()); } } catch (InstantiationException | IllegalAccessException e) { e.printStackTrace(); } } catch (ClassNotFoundException e) { e.printStackTrace(); } } } private void ioc() { for (String beanName : classList) { try { beanName = beanName.replace(".class", ""); Class beanClass = Class.forName(beanName); Field[] fields = beanClass.getDeclaredFields(); for (Field field : fields) { if (field.isAnnotationPresent(Resource.class)) { Resource resource = field.getAnnotation(Resource.class); String val = resource.value(); try { field.setAccessible(true); Service service = (Service) beanClass.getAnnotation(Service.class); if (service != null) { field.set(beanInstanceMap.get(service.value()), beanInstanceMap.get(val)); } else { Controller controller = (Controller) beanClass.getAnnotation(Controller.class); if (controller != null) { field.set(beanInstanceMap.get(controller.value()), beanInstanceMap.get(val)); } } } catch (IllegalAccessException e) { e.printStackTrace(); } } } } catch (ClassNotFoundException e) { e.printStackTrace(); } } } private void initUrlHandlerMap() { for (String beanName : classList) { try { beanName = beanName.replace(".class", ""); Class beanClass = Class.forName(beanName); if (beanClass.isAnnotationPresent(RequestMapping.class)) { RequestMapping baseReqMap = (RequestMapping) beanClass.getAnnotation(RequestMapping.class); if (baseReqMap != null) { String requestBaseUrl = baseReqMap.value(); Method[] methods = beanClass.getMethods(); for (Method method : methods) { RequestMapping reqMap = method.getAnnotation(RequestMapping.class); if (reqMap != null) { urlHandlerMap.put(requestBaseUrl + reqMap.value(), method); } } } } } catch (ClassNotFoundException e) { e.printStackTrace(); } } } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws javax.servlet.ServletException, IOException { doGet(request, response); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws javax.servlet.ServletException, IOException { String uri = request.getRequestURI(); String contextPath = request.getContextPath(); String requestUrl = uri.replace(contextPath, ""); Method method = urlHandlerMap.get(requestUrl); if (method == null) { return; } Class beanClass = method.getDeclaringClass(); Controller controller = (Controller) beanClass.getAnnotation(Controller.class); Object bean = beanInstanceMap.get(controller.value()); try { Map<String, String[]> requestParameterMap = request.getParameterMap(); Object[] args = new Object[requestParameterMap.entrySet().size()]; int index = 0; for (Map.Entry entry : requestParameterMap.entrySet()) { args[index] = entry.getValue(); index++; } method.invoke(bean, args); } catch (IllegalAccessException | InvocationTargetException e) { e.printStackTrace(); } } }

 

创建完servlet后web.xml自动生成相应代码

<load-on-startup>加这个标签,tomcat启动时会自动加载指定的servlet  大于等于0时启动tomcat会加载 servlet的init()方法 数字越小代表加载的优先级越高

<url-pattern>/这个标签是tomcat请求扫描路径,扫到后会调用servlet相应的doGet或doPost方法

 

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1">
    <servlet>
        <servlet-name>DispatcherServlet</servlet-name>
        <servlet-class>com.hongsen.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>DispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

</web-app>

创建Controller类

package com.hongsen.controller;

@Controller("testController")
@RequestMapping("/test")
public class TestController {

    @Resource("testAddService")
    private TestAddServiceImpl testAddService;

    @Resource("testDelService")
    private TestDelServiceImpl testDelService;

    @RequestMapping("/add.json")
    public void testAdd() {
        testAddService.add();
    }
    @RequestMapping("/del.json")
    public void testDel(Object num1, Object num2) {
        testDelService.del(num1.toString(), num2.toString());
    }
}

 创建service接口和实现类

package com.hongsen.service;

public interface TestAddService {
    void add(String num);
}

 

package com.hongsen.service;

@Service("testAddService")
public class TestAddServiceImpl implements TestAddService{
    @Override
    public void add(String num) {
        System.out.println("add方法执行!");
    }
}

 

部署tomcat启动成功后:

访问:http://localhost:8080/test/add.json

posted @ 2018-07-27 15:08  王小森#  阅读(398)  评论(0编辑  收藏  举报