Spring的笔记之手写一个简单的mvc框架

4 篇文章 0 订阅
1 篇文章 0 订阅

这是最近学习的一些整理,通过手写一个mvc的框架来认识和了解Spring的一些工作原理。
首先简单介绍一下Sprin的ioc容器,从本质上来说ioc容器就是一个以beanId为key,class为value的hashMap。Spring则通过配置文件和注解的方式将类放入到hashMap中。

再来,在mvc框架中,url和method的映射本质上也是一个hashMap,它的key是url,value是method。

介绍完以上这些基本的容器之后,接下来就是手写mvc框架的具体步骤
1.加载配置文件
2.解析配置文件并扫描所有的类
3.初始化所有的类并保存到ioc容器
4.完成自动化的依赖注入
5.建立method和url的映射

具体实现请看代码

public class LXADispathcerServlet extends HttpServlet {



    private Properties contextConfig = new Properties();

    private List<String> classNames = new ArrayList<>();

    private Map<String,Object> ioc = new HashMap<>();

    private Map<String,Method> handlerMapping = new HashMap<>();

    @Override
    public void init(ServletConfig config) throws ServletException {

        //1、加载配置文件
        doLoadConfig(config.getInitParameter("contextConfigLocation"));
        //2、解析配置文件,扫描所有相关的类
        try {
            doScanner(contextConfig.getProperty("scanPackage"));
        } catch (MalformedURLException e) {
            e.printStackTrace();
        }
        //3、初始化所有的相关的类,并保存到IOC容器中
        doInstance();
        //4、完成自动化的依赖注入
        doAutoWired();
        //5、创建HandlerMapping,将URL和Method建立对应关系
        initHandlerMapping();

        System.out.println(" Spring MVC is run");
    }

    private void doScanner(String scanPackage) throws MalformedURLException {

        URL url = this.getClass().getClassLoader().getResource("/" + scanPackage.replaceAll("\\.", "/"));
        URL ur2 = new URL(url.toString().replaceAll("/%5c","/"));
        File clssDir = new File(ur2.getFile());

        for (File file : clssDir.listFiles()) {

            //递归判断
            if (file.isDirectory()) {
                doScanner(scanPackage + "." + file.getName());
            } else {
                if (!file.getName().contains(".class")) {
                    continue;
                }
                //将所有相关的类放入容器中
                String className = scanPackage + "." + file.getName().replace(".class", "").trim();
                classNames.add(className);
            }
        }

    }


    private void doInstance() {
        if (classNames.isEmpty()) {
            return;
        }
        try {
            for (String className : classNames){
                Class<?> clazz = Class.forName(className);

                // 不是所有的类需要初始化,过滤
                if(clazz.isAnnotationPresent(LXAController.class) ){
                    String beanName = lowFirstCast(clazz.getSimpleName());
                    Object instance = clazz.newInstance();
                    ioc.put(beanName, instance);

                }else if(clazz.isAnnotationPresent(LXAService.class)){

                    //1.类名首字母小写
                    //2.自定义的名字(优先)

                    LXAService service = clazz.getAnnotation(LXAService.class);
                    String beanName = service.value();
                    if ("".equals(beanName.trim())){
                        //说明没有自定义名字,用默认首字母小写
                        beanName = lowFirstCast(clazz.getSimpleName());
                    }
                    Object instance = clazz.newInstance();
                    ioc.put(beanName,instance);
                    //3.用接口的全称作为key,用接口的实现类做为值
                    Class<?>[] interfaces = clazz.getInterfaces();
                    for (Class<?> i : interfaces){
                        if (ioc.containsKey(i.getName())){
                            throw new Exception("the beaName has exits");
                        }
                        ioc.put(i.getName(),instance);
                    }

                }else{
                    continue;
                }
            }
        }catch (Exception e){
            e.printStackTrace();
        }


    }

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

    private void doAutoWired() {

        if (ioc.isEmpty()){return;}
        for (Map.Entry<String, Object> entry : ioc.entrySet()) {
            Field[] declaredFields = entry.getValue().getClass().getDeclaredFields();

            for (Field field : declaredFields) {

                //只有注解了才初始化
                if (!field.isAnnotationPresent(LXAAutoWired.class)){continue; }

                LXAAutoWired autoWired = field.getAnnotation(LXAAutoWired.class);
                String beanName = autoWired.value();

                //使用了默认的名称
                if ("".equals(beanName)){
                    beanName = field.getType().getName();
                }

                //强制将不可见的属性可见化(private-》 public)
                field.setAccessible(true);
                try {
                    field.set(entry.getValue(),ioc.get(beanName));
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                    continue;
                }

            }
        }

    }

    private void initHandlerMapping() {

        if (ioc.isEmpty()){return;}

        for (Map.Entry<String, Object> entry : ioc.entrySet()) {

            Class<?> clazz = entry.getValue().getClass();

            if (!clazz.isAnnotationPresent(LXAController.class)){continue;}
            String beanUrl = "";
            if (clazz.isAnnotationPresent(LXARequestMapping.class)){
                LXARequestMapping requestMapping = clazz.getAnnotation(LXARequestMapping.class);
                beanUrl = requestMapping.value();

            }

            Method[] methods = clazz.getMethods();

            for (Method method : methods){
                if (!method.isAnnotationPresent(LXARequestMapping.class)){continue;}

                LXARequestMapping requestMapping = method.getAnnotation(LXARequestMapping.class);
                //用正则将多个斜杠替换为单个斜杠
                String url = ("/" + beanUrl + "/" + requestMapping.value()).replaceAll("/+","/");
                handlerMapping.put(url,method);

                System.out.println("Mapped:" + url + "," + method);
            }

        }

    }


    private void doLoadConfig(String config) {
        InputStream resourceAsStream = this.getClass().getClassLoader().getResourceAsStream(config);
        try {
            contextConfig.load(resourceAsStream);
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (null != resourceAsStream) {
                try {
                    resourceAsStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }

    }

    private void doDispather(HttpServletRequest req, HttpServletResponse resp) throws Exception {

        if (this.handlerMapping.isEmpty()){return;}
        String url = req.getRequestURI();
        String contextPath = req.getContextPath();
        url = url.replace(contextPath,"").replaceAll("/+","/");

        if (!this.handlerMapping.containsKey(url)){
            resp.getWriter().write("404 Not Found");
            return;
        }

        Method method = this.handlerMapping.get(url);

        Map<String, String[]> params = req.getParameterMap();
        Class<?> declaringClass = method.getDeclaringClass();
        String simpleName = declaringClass.getSimpleName();
        String beanName = lowFirstCast(simpleName);


        Object name = method.invoke(ioc.get(beanName), new Object[]{req, resp, params.get("name")[0]});
        resp.getWriter().write(name.toString());

    }

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doPost(req, resp);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //6.运行
        try {
            doDispather(req,resp);
        } catch (Exception e) {
            e.printStackTrace();
            resp.getWriter().write("500 Exception" + Arrays.toString(e.getStackTrace()));
        }
    }


}


项目地址 https://github.com/skiill/spring_mv_demo.git

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值