简单仿写MVC

代码地址(需要自取):mvc_Imitation: 简单仿写实现MVC (gitee.com)

项目目录

先把架子搭好

Controller注解

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

RequestMapping

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

HydController

@Controller
@RequestMapping("hyd")
public class HydController {
    @RequestMapping
    public  String index(){
        System.out.println("RequestMapping为“hyd, ”的方法已执行");
        return "";
    }
    @RequestMapping("hyd2")
    public  String index1(){
        System.out.println("RequestMapping为“hyd,hyd2”的方法已执行");
        return "";
    }
}

indexController

@Controller
@RequestMapping
public class IndexController {
    @RequestMapping
    public void index() {
        System.out.println("RequestMapping为“ , ”的方法已执行");
    }
}

Main

public class Main {
    static {
//        获取Main的路径
        String path = Main.class.getResource("").getPath();
//        获取Main的包名
        String packageName = Main.class.getPackage().getName();
        
        HydMVC.scanner(path, packageName);
    }

    public static void main(String[] args) {
        HydMVC.mehtod_go("","");
        HydMVC.mehtod_go("hyd","");
        HydMVC.mehtod_go("hyd","hyd2");
        HydMVC.mehtod_go("232323","23131");
        HydMVC.mehtod_go("hyd","23131");
    }

}

HydMVC

public class HydMVC {
//    存放 一级注解-二级注解-方法 的map
    private static HashMap<String,HashMap<String, Method>> map_method = new HashMap<>();
//    存放 一级注解-类的实例化对象 的map
    private static HashMap<String, Object> map_object = new HashMap<>();

    /**
     * 方法执行函数
     * @param first_path 第一路径
     * @param second_path 第二路径
     */
    public static void mehtod_go(String first_path,String second_path){
//        如果map_object中没有fisrt_path,则不存在该注解的类
        if (map_object.get(first_path)==null){
            System.out.println("没有一个类有"+first_path+"注解");
        }else {
            if (map_method.get(first_path).get(second_path)==null){
                System.out.println(first_path+"下没有"+second_path+"注解"+"的方法");
            }else{
                try {
                    map_method.get(first_path).get(second_path).invoke(map_object.get(first_path));
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                } catch (InvocationTargetException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    /**
     * 扫描获取类实例化对象以及方法
     * @param path
     * @param packageName 包名
     */

    public static void scanner(String path,String packageName){
//        扫描当前所有的文件
        List<String> paths = traverseFolder2(path);
//        遍历拿到的文件路径
        for (String path1 : paths) {
//              path1为:C:\ttt\imitating_spring_mvc\out\production\imitating_spring_mvc\com\heaboy\IndexController.class
//              先拿到文件名 例:IndexController.class
            path1=path1.substring(path.length()-1);
            try{
                //            构建类名
//            因为可能存在多级路径,例如:www/IndexController.class,所以还需要进行一些小处理,不能直接拼接
//            File.separator代表文件分隔符
//            Matcher.quoteReplacement确保了文件分隔符在替换过程中被正确地转义,以防包含任何正则表达式的特殊字符。它返回一个适合用于正则表达式替换的字符串。
                String ClassName = packageName + "." + path1.replaceAll(Matcher.quoteReplacement(File.separator),".");
//            去掉文件名中的.class后缀
                String load_className = ClassName.replace(".class","");
//            通过类加载器加载类
                Class<?> cl = ClassLoader.getSystemClassLoader().loadClass(load_className);
//            用写好的检查注解方法(isController)判断cl是否有Controller注解
                if(isController(cl)){
//              用写好的检查注解方法(isRequestMapping)判断cl是否有RequestMapping注解
                    if (isRequestMapping(cl)){
//                    如果有就获取类上RequestMapping注解的实例
                        RequestMapping requestMapping_first = getRequestMapping(cl);
//                    判断map_method中是否已经存在了该一级注解
                        if (map_method.containsKey(requestMapping_first.value())){
//                        已经存在该一级注解,则抛出异常
                            throw new RuntimeException("包含多个注解:"+requestMapping_first.value());
                        }else {
//                        不存在,添加至map_method中
                            map_method.put(requestMapping_first.value(),new HashMap<>());
//                        添加类的实例化对象到
                            map_object.put(requestMapping_first.value(),cl.newInstance());
                        }
//                    获取类中所有的方法
                        Method[] declareMethods = cl.getDeclaredMethods();
//                    遍历所有方法
                        for (Method method : declareMethods) {
                            if (isRequestMapping(method)){
                                RequestMapping requestMapping_second = getRequestMapping(method);
//                            判断该二级注解是否已经存在
                                if (map_method.get(requestMapping_first.value()).containsKey(requestMapping_second.value())){
//                                存在抛出错误
                                    throw new RuntimeException("方法注解已经存在:"+requestMapping_second.value());
                                }else {
//                                不存在就将方法放入map_method
                                    map_method.get(requestMapping_first.value()).put(requestMapping_second.value(),method);
                                }
                            }
                        }
                    }else{
                        throw new RuntimeException("该类有没RequestMapping注解");
                    }
                }
            }catch (Exception e){
                System.out.println(e);
            }
        }

    }



    /**
     * 扫描文件
     * @param path 绝对路径
     * @return List<String> 文件绝对路径集合
     * C:\ttt\imitating_spring_mvc\out\production\imitating_spring_mvc\com\heaboy\IndexController.class
     */
    private static List<String> traverseFolder2(String path) {
//        当前路径下的文件和文件夹
        File file = new File(path);
//        返回的文件集合
        List<String> file_list = new ArrayList<>();

        if (file.exists()){
//            放文件夹
            LinkedList<File> list = new LinkedList<>();

//            存储过程中要处理的文件
            File[] files = file.listFiles();

            for (File file1 : files) {
//                如果当前文件是文件夹就把它放到list里面
                if (file1.isDirectory()){
                    list.add(file1);
                }else { //不是就将该文件的绝对路径放到返回列表中file_list
                    file_list.add(file1.getAbsolutePath());
                }
            }
//            申请一个临时变量
            File file_temp;
//            当存放文件夹的队列不为空时,执行以下操作
            while (!list.isEmpty()){
//                取出第一个文件夹来处理
                file_temp=list.removeFirst();
//                使用上面申请的FIle[]来存放file_temp中的文件
                files=file_temp.listFiles();
//                处理files中的文件,同上,文件夹放到list中,文件放到file_list中
                for (File file1 : files) {
                    if (file1.isDirectory()){
                        list.add(file1);
                    }else {
                        file_list.add(file1.getAbsolutePath());
                    }
                }
            }
        }else{
//            如果没有任何文件,则不做处理
        }
        return file_list;
    }

//    判断类是否有Controller注解
    private static boolean isController(Class cl){
        Annotation annotation = cl.getAnnotation(Controller.class);
        if(annotation!=null){
            return  true;
        }
        return false;
    }

//    判断类是否有RequestMapping注解
    private static boolean isRequestMapping(Class cl){
        Annotation annotation = cl.getAnnotation(RequestMapping.class);
        if(annotation!=null){
            return  true;
        }
        return false;
    }
//    获取类的RequestMapping注解的实例
    private static RequestMapping getRequestMapping(Class<?> cl) {
        Annotation annotation = cl.getAnnotation(RequestMapping.class);
        if (annotation instanceof  RequestMapping){
            return (RequestMapping) annotation;
        }
        return null;
    }

    //    判断方法是否有RequestMapping注解
    private static boolean isRequestMapping(Method method) {
        Annotation annotation = method.getAnnotation(RequestMapping.class);
        if (annotation!=null){
            return true;
        }
        return false;
    }
//    获取方法上的RequestMapping注解的实例
    private static RequestMapping getRequestMapping(Method method) {
        Annotation annotation  = method.getAnnotation(RequestMapping.class);
        if (annotation instanceof RequestMapping){
            return (RequestMapping) annotation;
        }
        return null;
    }

}

整体思路流程,先对路径下的文件进行扫描,拿到.class后缀的文件,然后根据不同的注解进行不同的操作,将实例化对象和方法都存到map中,再从map中获取调用。具体详细的解释可以查看代码及其注释。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/786180.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

java设计模式(十一)组合模式(Composite Pattern)

1、模式介绍&#xff1a; 组合模式是一种结构型设计模式&#xff0c;它允许你将对象组合成树形结构以表示“部分-整体”的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。 2、应用场景&#xff1a; 表示树形结构&#xff1a;当你需要表示对象的部分-整体…

2024年06月CCF-GESP编程能力等级认证Python编程四级真题解析

本文收录于专栏《Python等级认证CCF-GESP真题解析》&#xff0c;专栏总目录&#xff1a;点这里&#xff0c;订阅后可阅读专栏内所有文章。 一、单选题&#xff08;每题 2 分&#xff0c;共 30 分&#xff09; 第 1 题 小杨父母带他到某培训机构给他报名参加CCF组织的GESP认证…

前端视角下的Spring-Boot语法学习:打印 hello-world

今日话题 基于 Spring Boot 打印输出 hello world 作者&#xff1a;云层上的光 时间&#xff1a;2024年6月20日 14时25分14秒 主线任务 一、打印 hello world 1、点击 “新建项目”用来演示 打印输出 “hello world” 2、填写项目配置&#xff1a;&#xff08;详细版见&a…

Python基础知识——(002)

文章目录 P8——7. input函数的使用 基本的输入函数input P9——8. Python中的注释 P10——9. Python中的缩进与本章总结 本章总结 P11——10. 章节习题 P8——7. input函数的使用 基本的输入函数input 语法结构&#xff1a; x input(提示文字) 注意事项&#xff1a;无论输…

【matlab】【python】爬虫实战

目录 引言 具体步骤 1.设置请求选项 2.发送请求并获取响应 3.设置正则表达式 4.执行正则表达式匹配 matlab完整代码 python代码示例 引言 在当今这个信息爆炸的时代&#xff0c;数据已成为推动社会进步和企业发展的核心动力之一。随着互联网的普及和技术的飞速发展&am…

前端视角下的Spring-Boot语法学习:demo-crud 实现增删改查

今日话题 基于 Spring Boot 实现增删改查&#xff0c;仅仅只是提供接口不涉及数据库增删改查 作者&#xff1a;云层上的光 时间&#xff1a;2024年6月21日 15时19分14秒 主线任务 一、项目创建 1、基于 idea 创建项目 2、选择项目依赖 Spring Web 二、实现增删改查 1、新…

Pix4Dmapper:无人机测绘的革命性工具

在现代测绘和地理信息系统&#xff08;GIS&#xff09;领域&#xff0c;Pix4Dmapper无疑是一款革命性的工具。作为一名长期使用这款软件的用户&#xff0c;我深深感受到它在工作中的重要性和便利性。Pix4Dmapper不仅仅是一款软件&#xff0c;更是测绘工作者的得力助手&#xff…

Selenium 的基本操作你知道哪些?

1. 前言 今天的推文&#xff0c;我们就来说说看&#xff0c;怎么实现模拟真人去打开微信读书网站。 2.需求分析和准备 整体的需求大致可以分为以下步骤&#xff1a; 打开chrome浏览器 打开百度网页 搜索“微信读书” 点击进入“微信读书”官网 搜索关键词“长安的荔枝” 点…

2023.2版IDEA复制配置修改端口增加一个当前运行服务的操作流程

文章目录 前言操作流程截图 前言 在微服务技术学习中很多学习场景会使用到运行多个服务节点进行调试&#xff0c;想要去模拟集群部署&#xff0c;就需要去复制配置&#xff0c;本文讲解一下如何复制&#xff0c;以及修改端口号。 操作流程截图

现代化木工装备建设新颖校园木工创客室

校园木工创客室是一个集木工制作、创意设计、科技融合与教育实践于一体的多功能空间。它为学生提供了一个动手实践、创新创造的平台&#xff0c;旨在培养学生的动手能力、创新思维、解决问题的能力以及团队协作能力。 木工创客室的设备选择应综合考虑需求、预算、品牌、质量、安…

Git常用技能速成

文章目录 一.版本控制二.提交并推送代码三.提交推送代码 一.版本控制 接下来&#xff0c;我们就需要对我们的功能进行优化&#xff0c;但是需要说明的是&#xff0c;我们不仅仅要对上述提到的缓存进行优化&#xff0c;还需要对我们程序的各个方面进行优化。我们本章节主要是针…

vue3在defineProps中使用多语言t,打包报错

报错原因 代码如下 打包后就会报错 defineProps() in script setup cannot reference locally declared variables because it will be hoisted outside of the setup() function. If your component options require initialization in the module scope, use a separate no…

自定义波形图View,LayoutInflater动态加载控件保存为本地图片

效果图: 页面布局: <?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="…

关于 RK3588刷镜像升级镜像”没有发现设备“ 的解决方法

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/140287339 长沙红胖子Qt&#xff08;长沙创微智科&#xff09;博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV…

js吸顶导航

吸顶导航 当我们浏览页面篇幅较大&#xff0c;浏览过半的时候想回到导航位置&#xff0c;只能通过往回滚动或通过”回到顶部”重新滚动到导航位置&#xff0c;这样的操作显得繁琐与不便。于是便有了吸顶式导航的交互方式&#xff0c;吸顶条导航最大的好处是将最常用或者设计者最…

命名空间namespace--c++入门基础等

个人主页点这里~ 1.命名空间-namespace 简介 &#xff1a;在C/C中&#xff0c;变量、函数和后面要学到的类都是大量存在的&#xff0c;这些变量、函数和类的名称将都存在于全局作用域中&#xff0c;可能会导致很多冲突。使用命名空间的目的是对标识符的名称进行本地化&#xf…

DataExcelServer局域网文件共享服务器增加两个函数

1、PFSUM合并指定路径下单元格ID的值 PFSUM("/103采购/8月采购名细","amount") 第一个参数为路径&#xff0c;第二个参数为单元格的ID 2、PFQuery 查询路径下 单元格ID值的列表 PFQuery("/103采购/8月采购名细","amount") 查询/103采…

在Mac上一键安装Mysql(解决所有安装问题)

重点强调安装mysql成功的关键在于安装的版本不能是最新&#xff01;&#xff01; 目录 一&#xff1a;下载mysql数据库安装部分到此结束 二&#xff1a;配置mysql数据库三&#xff1a;启动mysql数据库四&#xff1a;各类奇葩问题总结 一&#xff1a;下载mysql数据库 1.进入MyS…

高清图片压缩无水印小程序源码系统 前后端分离 带完整的安装代码包以及搭建教程

系统概述 在当今的数字化时代&#xff0c;图片作为信息传播的重要载体&#xff0c;其质量和传输效率直接影响到用户体验。然而&#xff0c;高清图片往往伴随着较大的文件体积&#xff0c;这不仅会占用大量存储空间&#xff0c;还会拖慢网页或应用的加载速度。因此&#xff0c;…

如何为IP申请SSL证书

目录 以下是如何轻松为IP地址申请SSL证书的详细步骤&#xff1a; 申请IP证书的基本条件&#xff1a; 申请IP SSL证书的方式&#xff1a; 确保网络通信安全的核心要素之一&#xff0c;是有效利用SSL证书来加密数据传输&#xff0c;特别是对于那些直接通过IP地址访问的资源。I…