IoC容器的设计(利用反射、注解和工厂模式实现)

news/2024/11/27 8:24:14/

1.实验要求

  1. 利用注解、反射和工厂模式设计一个简单的IoC容器
  2. 该IoC容器包含3个注解和一个IoC容器类(AnnotationConfigApplicationContext),其定义如下:

注解:

注解含义
@Component标注Bean
@Autowired标注需要被注入的对象
@Configuration标注为配置类
@ComponentScan注解扫描器

IoC容器类:

        3.自定义两个业务类Group和User,创建一个测试类Test,对IoC容器进行测试:

2.实验思路

本实验要求我们通过反射和自定义注解的设计,模拟实现spring框架IoC容器类的基础功能,要实现的注解有@Component、@Autowired、@Configuration和@ComponentScan

首先我们需要明确这些注解在spring框架中本身的功能是什么:

  1. @Component:将实体类实例化到容器中
  2. @ComponentScan:配置包路径,到该路径下去寻找bean
  3. @Autowired:自动装配(定义一个变量接收注入的类)
  4. @Configuration:标识为配置类

所以在该实验中,我们需要将自定义四个注解,然后将Group和User类使用@Component注解,在User类中创建Group类的实例化对象并设置为自动装配,这样就能在User类中调用Group类的方法;

然后我们需要自己实现一个IoC容器类,处理自定义的注解的基本逻辑;

接下来在test类中实例化IoC容器,并从中取得bean,调用其方法

3.实验代码

pom文件依赖配置:

<dependencies><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.2.10.RELEASE</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>5.2.14.RELEASE</version></dependency><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><scope>test</scope></dependency>

jdk版本:1.8

开发工具:IDEA

3.1自定义注解

@Autowired:

//自定义注解:标注需要被注入的对象
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Autowired {
}

@Component:

//自定义注解:标注Bean
@Target({ElementType.TYPE,ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Component {String value() default "";
}

@ComponentScan:

//自定义注解 注解扫描器
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface ComponentScan {String value() default "";
}

@Configuration:

//自定义注解 标注为配置类
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Configuration {
}

3.2 定义实体类

Group:

@Component
public class Group {public String getBeanName(){return this.getClass().getSimpleName();}
}

User:

@Component
public class User {@Autowiredprivate Group group;public void print(){System.out.println("group name as follow:");System.out.println(group.getBeanName());}}

3.3 定义配置类

MyConfig:

@Configuration
@ComponentScan("org.example")
public class MyConfig {
}

3.4 定义IoC容器类

接口:ApplicationContext

public interface ApplicationContext {Object getBean(Class clazz);
}

实现类:AnnotationApplicationContext

public class AnnotationApplicationContext implements ApplicationContext {//定义Map,用于存放bean对象private Map<String, Object> beanFactory = new HashMap<>();//创建IOC容器类时执行的方法public AnnotationApplicationContext(Class configClass) throws InstantiationException, IllegalAccessException {//处理@Configuration的逻辑(没有实质性的内容,只是要求必须有配置类)try {//获取@Configuration类型的对象Configuration configuration = (Configuration) configClass.getAnnotation(Configuration.class);//如果对象为空,则引发异常if (configuration.annotationType().toGenericString().equals("")) {//没有具体逻辑}} catch (Exception e) {System.out.println("找不到配置类");System.exit(-1);}//处理@ComponentScan的逻辑/*总的来说,就是扫描某一路径下的所有类,如果有的类被Component注解,则创建其实例(一个bean),加入beanFactory中如果有的类中的对象(一个Field类型的数据)被@Autowired注解标注,则用beanFactory中同类型的bean替代该对象*/try {ComponentScan componentScan = (ComponentScan) configClass.getAnnotation(ComponentScan.class);//根据componentScanValue的值来获取要扫描的路径(默认值处理的不是很好)String componentScanValue = componentScan.value().toString();String path = "";if (!componentScanValue.equals("")){String[] splitValue = componentScanValue.split("\\.");for (int i = 0; i < splitValue.length; i++) {path += splitValue[i] + "/";}path = "classpath*:" + path + "**/*.class";}else {path = "classpath*:org/example/**/*.class";}//扫描路径,获取所有的classPathMatchingResourcePatternResolver resolver = new PathMatchingResourcePatternResolver();Resource[] resources = resolver.getResources(path);for (Resource res : resources) {//获取类名String clsName = new SimpleMetadataReaderFactory().getMetadataReader(res).getClassMetadata().getClassName();//获取类对象Class thisClass = Class.forName(clsName);//判断当前类是否被Component注解标注Component component = (Component) thisClass.getAnnotation(Component.class);if (component != null) //说明该类被Component注解标注{Constructor c = thisClass.getConstructor();Object o = c.newInstance(); //创建该类的实例beanFactory.put(o.getClass().getSimpleName(), o);//加入beanFactory中}//处理@Autowired注解for (Object bean : beanFactory.values()) { //查看beanFactory中所有的beanClass beanClass = bean.getClass();Field[] beanFields = beanClass.getDeclaredFields(); //获取bean的fieldsfor (Field field : beanFields) {if (field.isAnnotationPresent(Autowired.class)) { //如果被Autowired注解标注Object beanD = beanFactory.get(field.getType().getSimpleName()); //获取beanFactory中的beanfield.setAccessible(true); //关闭安全检查field.set(bean, beanD); //用beanFactory中的bean来代替当前bean}}}}} catch (Exception e) {e.printStackTrace();System.out.println("没有ComponentScan注解");System.exit(-1);}}//getBean方法,获取某个bean(通过class)@Overridepublic Object getBean(Class beanClass) {return beanFactory.get(beanClass.getSimpleName());}}

3.5 测试类

public class test {public static void main(String[] args) throws InstantiationException, IllegalAccessException {//创建IOC容器,传入的参数是当前的配置类//因此可以根据@Configuration和@ComponentScan("org.example")来走接下来的逻辑AnnotationApplicationContext context = new AnnotationApplicationContext(MyConfig.class);//从容器中获得user对象User user = (User) context.getBean(User.class);//执行对象方法//可以看到通过user对象执行了group类的方法getBeanName()//但user类中,我们并没有实例化group对象,可以看到是通过注解来实现的user.print();}
}

4.实验结果

成功在user类中调用了group类的方法;

5.源码

上面的代码是完整的,下附工程文件:

https://download.csdn.net/download/qq_51235856/87944324


http://www.ppmy.cn/news/553493.html

相关文章

rust abc(1): 最小环境搭建

文章目录 1. 目的2. 命令集合3. 安装或更新 rust3.1 命令3.2 运行结果 4. 包管理工具 Cargo5. 创建 Rust 的 Hello World 程序: 单个文件6. 创建 Rust 的 Hello World 工程&#xff1a; 基于 Cargo6.1 cargo new 创建工程6.2 cargo run6.3 完整输出6.4 解释 7. IDE/编辑器8. Re…

悄悄关注 (25分)

题目描述 新浪微博上有个“悄悄关注”&#xff0c;一个用户悄悄关注的人&#xff0c;不出现在这个用户的关注列表上&#xff0c;但系统会推送其悄悄关注的人发表的微博给该用户。现在我们来做一回网络侦探&#xff0c;根据某人的关注列表和其对其他用户的点赞情况&#xff0c;扒…

7-5 悄悄关注

悄悄关注 新浪微博上有个“悄悄关注”&#xff0c;某人人悄悄关注的人&#xff0c;不出现在某人人的关注列表上&#xff0c;但系统会推送其悄悄关注的人发表的微博给某人人。现在我们来做一回网络侦探&#xff0c;根据某人人的关注列表和其对其他用户的点赞情况&#xff0c;扒…

L2-019 悄悄关注

新浪微博上有个“悄悄关注”&#xff0c;一个用户悄悄关注的人&#xff0c;不出现在这个用户的关注列表上&#xff0c;但系统会推送其悄悄关注的人发表的微博给该用户。现在我们来做一回网络侦探&#xff0c;根据某人的关注列表和其对其他用户的点赞情况&#xff0c;扒出有可能…

7-5 悄悄关注 (25 分)

新浪微博上有个“悄悄关注”&#xff0c;一个用户悄悄关注的人&#xff0c;不出现在这个用户的关注列表上&#xff0c;但系统会推送其悄悄关注的人发表的微博给该用户。现在我们来做一回网络侦探&#xff0c;根据某人的关注列表和其对其他用户的点赞情况&#xff0c;扒出有可能…

Redis实现微博好友功能微服务(关注,取关,共同关注)

需求分析 好友功能是目前社交场景的必备功能之一&#xff0c;一般好友相关的功能包含有&#xff1a;关注/取关、我&#xff08;他&#xff09;的关注、我&#xff08;他&#xff09;的粉丝、共同关注、我关注的人也关注他等这样一些功能。 类似于这样的功能我们如果采用数据库…

7-7 悄悄关注 (25分)

7-7 悄悄关注 (25分) 新浪微博上有个“悄悄关注”&#xff0c;一个用户悄悄关注的人&#xff0c;不出现在这个用户的关注列表上&#xff0c;但系统会推送其悄悄关注的人发表的微博给该用户。现在我们来做一回网络侦探&#xff0c;根据某人的关注列表和其对其他用户的点赞情况&…

我的关注

欢迎使用Markdown编辑器写博客 本Markdown编辑器使用StackEdit修改而来&#xff0c;用它写博客&#xff0c;将会带来全新的体验哦&#xff1a; Markdown和扩展Markdown简洁的语法代码块高亮图片链接和图片上传LaTex数学公式UML序列图和流程图离线写博客导入导出Markdown文件丰…