自己编写Spring mvc简易框架

学习java web开发的小伙伴,对spring MVC 不会陌生的。最近没事看了一下spring的源码,了解了一些框架的实现方法,再结合网上一些资料,自己编写了一个简易的MVC框架,现将记录下开发工程。
一、首先,我们先来介绍一下Spring的三个阶段,配置阶段、初始化阶段和运行阶段
配置阶段:主要是完成application.properties配置和Annotation配置。

初始化阶段:主要是加载并解析配置信息,然后,初始化IOC容器,完成容器的DI操作,已经完成HandlerMapping的初始化。

运行阶段:主要是完成Spring容器启动以后,完成用户请求的内部调度,并返回响应结果。
二、配置阶段
我采用的是maven管理项目。先来看pom.xml文件中的配置,只引用了servlet-api的依赖。


      javax.servlet
      javax.servlet-api
      3.0.1
      provided
    

在web.xml文件中配置以下信息:


    
        spring MVC
        com.xiao.servlet.MyDispatcherServlet
        
            contextConfigLoaction
            application.properties
        
        1
    
    
        spring MVC
        /*
    

application.properties 配置

 scanPackage=com.xiao.demo

接下来,配置注解。
创建MyController 注解:

package com.xiao.annotation;

import java.lang.annotation.*;

/**
 * 注解在类上
 */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyController {
    String value() default "";
}

创建MyService 注解:

package com.xiao.annotation;

import java.lang.annotation.*;

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

创建MyRequestMapping注解:

package com.xiao.annotation;

import java.lang.annotation.*;

/**
 * 注解在类和方法上
 */
@Target({ElementType.TYPE,ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyRequestMapping {
    String value() default "";
}

创建MyAutowired注解:

  package com.xiao.annotation;
   import java.lang.annotation.*;
    @Target(ElementType.FIELD)
    @Retention(RetentionPolicy.RUNTIME)
    @Documented
    public @interface MyAutowired {
        String value() default "";
    }

创建MyRequestParam注解:

   package com.xiao.annotation;

import java.lang.annotation.*;

@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyRequestParam {
    /**
     * 表示参数的别名,必填
     * @return
     */
    String value();
}

三、初始化、运行阶段
创建GPDispatcherServlet类并继承HttpServlet,重写init()、doGet()和doPost()方法。

package com.xiao.servlet;

import com.xiao.annotation.MyAutowired;
import com.xiao.annotation.MyController;
import com.xiao.annotation.MyRequestMapping;
import com.xiao.annotation.MyService;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.URL;
import java.util.*;

/**
 * @author xiaoxiaoping
 */
public class MyDispatcherServlet  extends HttpServlet {

    private static final long serialVersionUID = 1L;

    /**
     * 跟web.xml中param-name的值一致
     */
    private static final String LOCATION = "contextConfigLoaction";

    /**
     * 保存所有的配置信息
     */
    private Properties properties = new Properties();

    /**
     * 核心IOC容器,保存所有初始化的bean
     */
    private Map ioc = new HashMap();

    /**
     * 保存所有的URL和方法的映射关系
     */
    private Map handelerMapping = new HashMap();

    /**
     * 保存所有臊面到的相关类名
     */
    private List classNames = new ArrayList();

    public MyDispatcherServlet() { super(); }

    /**
     * 初始化,加载配置文件
     * @param config
     * @throws ServletException
     */
    @Override
    public void init(ServletConfig config) throws ServletException {

        //加载配置文件
        doLoadConfig(config.getInitParameter(LOCATION));

        //扫描所有相关类
        doScanner(properties.getProperty("scanPackage"));

        //初始化所有相关类的实例,并保存到IOC容器
        doInstance();

        //依赖注入
        doAutowired();

        //构造HandleMapping
        initHandlerMapping();

        //等待请求,匹配URL,定位方法,反射调用执行
        //调用doGet或者doPost方法

        //提示信息
        System.out.println("xiaoMvc is init");
    }

    private void initHandlerMapping() {
        if(ioc.isEmpty()){return;}
        for(Map.Entry entry : ioc.entrySet()){
            Class  clazz = entry.getValue().getClass();
            if(!clazz.isAnnotationPresent(MyController.class)){continue;}
            String baseUrl = "";
            //获取controller的url配置
            if(clazz.isAnnotationPresent(MyRequestMapping.class)){
                MyRequestMapping requestMapping = clazz.getAnnotation(MyRequestMapping.class);
                baseUrl = requestMapping.value();
            }

            Method [] methods = clazz.getMethods();
            for(Method method : methods){
                if(!method.isAnnotationPresent(MyRequestMapping.class)){continue;}
                MyRequestMapping requestMapping = method.getAnnotation(MyRequestMapping.class);
                String url = ("/"+baseUrl+"/"+requestMapping.value()).replaceAll("/+", "/");
                handelerMapping.put(url,method);
                System.out.println("mapped"+url+" , "+method);
            }

        }
    }

    private void doAutowired() {
        if(ioc.isEmpty()){return;}
        for(Map.Entry entry : ioc.entrySet()){
            Field[] fields = entry.getValue().getClass().getDeclaredFields();
            for(Field field : fields){
                if(!field.isAnnotationPresent(MyAutowired.class)){continue;}
                MyAutowired autowired = field.getAnnotation(MyAutowired.class);
                String beanName = autowired.value();
                if("".equals(beanName.trim())){
                    beanName = lowerFirstCase(field.getType().getSimpleName());
                }
                //设置私有属性的访问权限
                field.setAccessible(true);
                try {
                    field.set(entry.getValue(),ioc.get(beanName));
                } catch (Exception e) {
                    e.printStackTrace();
                    continue;
                }
            }
        }
    }

    private void doInstance() {
        if(classNames.size()==0){return;}
        try {
            for(String className : classNames) {
                Class clazz = Class.forName(className);
                if(clazz.isAnnotationPresent(MyController.class)){
                    //默认将首字母小写作为beanName
                    String beanName = lowerFirstCase(clazz.getSimpleName());
                    ioc.put(beanName,clazz.newInstance());
                }else if(clazz.isAnnotationPresent(MyService.class)){
                    MyService service = clazz.getAnnotation(MyService.class);
                    String beanName = service.value();
                    //如果用户设置了名字,就用用户自己设置的
                    if("".equals(beanName)){
                        beanName = lowerFirstCase( clazz.getSimpleName() );
                    }
                    ioc.put(beanName,clazz.newInstance());
                }else{
                    continue;
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    private void doScanner(String scanPackage) {

        URL url = this.getClass().getClassLoader().getResource("/"+scanPackage.replaceAll("\\.","/"));
        File dir = new File(url.getFile());
        for(File file : dir.listFiles()){
            if(file.isDirectory()){
                doScanner(scanPackage+"."+file.getName());
            }else{
                classNames.add(scanPackage+"."+file.getName().replace(".class", ""));
            }
        }

    }

    private void doLoadConfig(String location) {
        //把web.xml中的contextConfigLocation对应value值的文件加载到流里面
        InputStream resourceAsStream = this.getClass().getClassLoader().getResourceAsStream(location);
        try {
            //用Properties文件加载文件里的内容
            properties.load(resourceAsStream);
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            //关流
            if(null!=resourceAsStream){
                try {
                    resourceAsStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    @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 {
        try {
            doDispatch(req,resp);
        } catch (Exception e) {
            e.printStackTrace();
            resp.getWriter().write("500!! Server Exception");
        }
    }

    private void doDispatch(HttpServletRequest req, HttpServletResponse resp) throws Exception {
        if(this.handelerMapping.isEmpty()){ return; }
        String url = req.getRequestURI();
        String contextPath = req.getContextPath();
        url = url.replace(contextPath,"").replaceAll("/+","/");
        if(!this.handelerMapping.containsKey(url)){
            resp.getWriter().write("404 not Found");
            return;
        }
        Method method = this.handelerMapping.get(url);
        Class[] parameterTypes = method.getParameterTypes();
        //获取请求的参数
        Map parameterMap = req.getParameterMap();
        //保存参数值
        Object[] paramValues = new Object[parameterTypes.length];
        for(int i = 0;i < parameterTypes.length; i++){
            Class parameterType = parameterTypes[i];
            if(parameterType == HttpServletRequest.class){
                paramValues[i] = req;
                continue;
            }else if(parameterType == HttpServletResponse.class){
                paramValues[i] = resp;
                continue;
            }else if(parameterType == String.class){
                for(Map.Entry param : parameterMap.entrySet()){
                    String value =Arrays.toString(param.getValue()).replaceAll("\\[|\\]", "").replaceAll(",\\s", ",");
                    paramValues[i]=value;
                }
            }
        }
        try {
            String beanName = lowerFirstCase(method.getDeclaringClass().getSimpleName());
            method.invoke(this.ioc.get(beanName), paramValues);
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            System.out.println("此处接收被调用方法内部未被捕获的异常");
            Throwable t = e.getTargetException();// 获取目标异常
            t.printStackTrace();
        }

    }

    private String lowerFirstCase(String str){
        char [] chars = str.toCharArray();
        chars [0] += 32;
        return String.valueOf(chars);
    }

}

四、测试实例

service类

    package com.xiao.demo.service.impl;
    
    import com.xiao.annotation.MyService;
    
    @MyService
    public class TestServiceImpl{
        public String query(String name, String age) {
            return name+"年龄"+age;
        }
    }
controller类
package com.xiao.demo;

import com.xiao.annotation.*;
import com.xiao.demo.service.impl.TestServiceImpl;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

@MyController
@MyRequestMapping("/test")
public class HelloController {
    @MyAutowired
    TestServiceImpl testService;

    @MyRequestMapping("/hello")
    public void test1(HttpServletRequest request, HttpServletResponse response,
                      @MyRequestParam("param") String param){
        System.out.println( testService.query("xiaoping","30"));

        try {
            response.getWriter().write(  " param:"+param);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

至此,一个简单的spring mvc 简易框架完成。
在此给出源码地址:https://gitee.com/longsis/xiaoMVC.git

你可能感兴趣的:(自己编写Spring mvc简易框架)