java自定义注解实现IOC

自定义注解实现简单的IOC

IOC

一般实现IOC使用的是java的反射技术,IOC和反射的概念再次不做赘述

1.首先我们需要自定义注解

本次是对ioc容器的简单实现,所以定义了3个用的最多的注解

Autowired ,依赖注入

import java.lang.annotation.*;

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Autowired {

}

Component ,生成bean交给容器

import java.lang.annotation.*;

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

public @interface Component {

    Class <?> classes ();

}

Value ,给属性赋值

import java.lang.annotation.*;

@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Value {

    String stringValue () default "";

    int intValue() default 0;

}

创建学生类,并添加自定义注解

@Component(classes = Student.class)
@Data
public class Student {


    @Value(stringValue = "jack")
    private String name;

}

接下来就是我们的简答实现步骤

2.添加pom依赖,该包是一个反射框架,这里我们用于扫描包

      <dependency>
            <groupId>org.reflections</groupId>
            <artifactId>reflections</artifactId>
            <version>0.9.11</version>
      </dependency>

3.有人说spring容器就是多个Map的有序结合,这里我们创建一个hashMap用于存储实例化后的对象

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

4.扫描包,找到所有需要实例化的对象,也就是有注解Component的类

   // 要扫描的包
   String packageName = "com.mytest.myIoc";
   Reflections ref = new Reflections(packageName);

   // 获取包下所有有注解的对象的class
   Set<Class<?>> classSet = ref.getTypesAnnotatedWith(Component.class);

5.实例化bean,通过反射,对有注解Value的字段进行赋值

 classSet.forEach(c -> {
            try {
                // 生成对象
                Object obj = c.newInstance();
                // 获取对象中的所有字段
                Field[] declaredFields = obj.getClass().getDeclaredFields();
                for (Field declaredField : declaredFields) {
                    // 设置私有属性可以访问
                    declaredField.setAccessible(true);
                    if (declaredField.isAnnotationPresent(Value.class)) {
                        String type = declaredField.getType().getName();
                        Object v = null;
                        if (type.equals("java.lang.String")) {
                            v = declaredField.getAnnotation(Value.class).stringValue();
                        }else if (type.equals("java.lang.Integer")){
                            v =declaredField.getAnnotation(Value.class).intValue();
                        }
                        declaredField.set(obj,v);
                    }
                }
                registerBean(c.getName(),obj);
            }catch (Exception e){
                log.error("对象生成失败");
                e.printStackTrace();
            }
  });

	 /**
     * 注册bean
     */
    private void registerBean(String name,Object obj){
        beanFactory.put(name,obj);
    }

6.到这里已经把实例好的对象交给map容器了,不妨打印一下map看看是否对象创建成功,字段是否赋值成功
在这里插入图片描述
可以看到对象已经创建成功,并且给name赋值成功

7.接下来就是实现依赖注入的功能了,首先我们定义一个私有属性,通过类型找到map容器中已经实例好的对象
注入

  @Autowired
  private Student student;
/**
     * 自动注入实现
     */
   private  void inject(Object o)  {
       Field[] declaredFields = o.getClass().getDeclaredFields();
       for (Field declaredField : declaredFields) {
           declaredField.setAccessible(true);
           if (declaredField.isAnnotationPresent(Autowired.class)){
               Class<?> type = declaredField.getType();
               Object bean = beanFactory.get(type.getTypeName());
               try {
                   declaredField.set(o,bean);
               } catch (IllegalAccessException e) {
                   throw new RuntimeException(e);
               }
           }
       }
   }

8.开始测试

 public static void main(String[] args) {
        // 实例化容器对象
        MyIoc myIoc = new MyIoc();
        myIoc.scanPackages();

        // 容器中的对象
        myIoc.beanFactory.forEach((k,v) ->{
            System.out.println(k + "->" + v);
        });

        // 此时为null
        System.out.println(myIoc.student);

        //实现注入
        myIoc.inject(myIoc);

        // 此时为Student(name=jack)
        System.out.println(myIoc.student);

   }

在这里插入图片描述

Spring框架自定义注解可以用于标记那些需要特殊处理或者额外配置的Bean。要使用自定义注解来修改IoC容器的Bean,你需要按照以下步骤操作: 1. **创建自定义注解**: - 定义一个新的注解接口,例如`@CustomBean`,并添加必要的属性,如果需要,可以提供默认值或类型。 ```java import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target; @Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) public @interface CustomBean { String name() default ""; // 添加其他属性... } ``` 2. **应用注解到Bean上**: - 在Spring组件(如@Configuration、@Service、@Controller等)类或单个Bean上使用这个注解。 ```java @Configuration @CustomBean(name = "customBean") public class CustomConfig { // Bean定义... } ``` 3. **处理注解**: - 创建一个`BeanDefinitionRegistryPostProcessor`来扫描和解析注解,对匹配的Bean做相应的修改。你可以通过覆盖`postProcessBeanDefinitions`方法实现这一点。 ```java @Component public class CustomBeanDefinitionProcessor implements ImportAware, BeanDefinitionRegistryPostProcessor { private static final String CUSTOM_BEAN_ATTRIBUTE_NAME = "customAttributeName"; @Override public void postProcessBeanDefinitions(Map<String, Object> beanDefinitions, BeanDefinitionRegistry registry) throws BeansException { for (String beanName : registry.getBeanDefinitionNames()) { if (isCustomBean(beanName)) { // 判断是否应用了CustomBean注解 ConfigurableListableBeanFactory factory = ((DefaultListableBeanFactory) registry).getBeanFactory(); GenericBeanDefinition definition = (GenericBeanDefinition) beanDefinitions.get(beanName); definition.getPropertyValues().addPropertyValue(CUSTOM_BEAN_ATTRIBUTE_NAME, getCustomValueFromAnnotation(beanName)); // 获取自定义属性并设置 } } } private boolean isCustomBean(String beanName) { return beanName.matches("@CustomBean.*"); } // ...其他获取自定义注解信息的方法 } ``` 4. **访问自定义属性**: - 现在,你可以在依赖注入的地方通过自定义属性来访问修改后的Bean。 ```java @Service public class MyService { @Autowired @CustomBean(name = "customBean", customAttributeName = "desiredValue") // 使用自定义属性 private MyCustomBean customBean; // ... } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

do{a++b++}while(a&b)

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值