学而不思则罔,思而不学则殆

前言

大家都用过Spring的@Value("xxx")注解,如果没有debug过源码的同学对这个操作还是一知半解,工作一年了学了反射学了注解,还是不会自己手撸一个注解对属性赋值的操作。今天就用几分钟时间给你讲明白这个如何实现!

理想中代码:

@Compant
public class Bean01 {
@MyValue("张三") //自定义注解
String name;
}

如果学过反射,获取类属性上面的自定义注解对象简直太简单,那怎么拿到“张三”,并给Bean01这个对象的name赋值呢?

在这里我用spring的形式给大家展示一下,完成这个理想赋值的demo~

思路:

1.spring启动,通过ComponentScan扫描注解(标签)加载@Component装饰的所有bean对象

2.通过Spring的BeanFactory增强,拿到Spring中注册的类信息

(BeanFactory会把扫描到的类信息放到BeanDefinitionMap中

BeanFactory会把扫描到的类名称放到BeanDefinitionNames中)

3.获取BeanDefinition中class信息,通过反射技术,获取类的属性,进而判断有没有自定义的注解装饰。

4.使用InvocationHandler,拿到自动义注解的属性值(memberValues : name=“张三”)

5.再通过Class使用反射创建对象,并进行类属性赋值

6.把赋值后对象注册到Spring容器中,会添加到Spring的一级缓存

7.进行对象获取的时候(getBean(xxx)),直接会从一级缓存中获取。这样就完成了我们注解赋值的操作~

代码实现

1.加载spring.xml

首先在/resources目录下创建spring.xml,目的是开启对Spring的注解支持

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
<context:component-scan base-package="com.*"/>
</beans>

2.创建自定义注解

在这里为了更好的演示,创建两个自定义注解类

@Target(ElementType.FIELD) // 装饰在类的属性上
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyValue {
String value() default "";//给姓名赋值用
}
@Target(ElementType.FIELD) // 装饰在类的属性上
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyValue2 {
int age() default 0; //给年龄赋值用
}

3.创建Bean对象

然后我在这里创建两个Bean的对象,使用@Compent进行注入到spring容器,这样在BeanFactory中就可以获取到对象信息(BeanDefinition)

@Component
public class Bean01 {
@MyValue("张三")
public String name; @MyValue2(age = 11)
public int age;
}
@Component
public class Bean02 {
@MyValue("李四")
public String name; @MyValue2(age = 18)
public int age;
}

4.使用PostProcessor进行扩展(逻辑在第6步)

启动Spring的时候,@Component所装饰的Bean对象如果实现了BeanDefinitionRegistryPostProcessor这个类,会在执行时显示地调用他的两个方法:

postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry)

postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory)

二者虽然都调用,但是区别在于其参数。**这里不过多介绍,可以在其他小伙伴的文章中进行学习*

@Component
public class MyPostProcess implements BeanDefinitionRegistryPostProcessor {
@Override
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {}
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {}
}

5.添加PostProcessor处理逻辑


@Component
public class MyPostProcess implements BeanDefinitionRegistryPostProcessor {
// 用来存放获取到存在注解配置的Bean对象
private Map<Class, Object> beanClassAndObjectMap;
// 解析完带有自定义注解的bean class
private Set<Class> hasMyAnnotationObjects;
// 用来存放当前操作的class对象
private Class currentClass; @Override
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
// 拿到BeanDefinition集合并对其初始化准备操作
initBeanClassAndObjectMap(registry, registry.getBeanDefinitionNames());
// 开始对包含自定义注解的属性进行赋值
executeSetField();
} // 此方法任务是从Spring的Feactory中,拿到所有的class信息,并通过反射进行实例化 存入到map集合中
private void initBeanClassAndObjectMap(BeanDefinitionRegistry registry, String[] beanDefinitionNames) {
// 初始化存在注解的BeanDefinition
beanClassAndObjectMap = new HashMap<>(beanDefinitionNames.length);
// 初始化用来可解析的Class容器
hasMyAnnotationObjects = new HashSet<>(beanDefinitionNames.length); // 根据beanDefinitionNames进行获取BeanDefinition
for (String beanDefinitionName : beanDefinitionNames) {
// 这个BeanDefinition包含了某个Bean对象的class信息
BeanDefinition beanDefinition = registry.getBeanDefinition(beanDefinitionName); // 如果这个Bean的定义信息是被注解修饰过的
if (beanDefinition instanceof ScannedGenericBeanDefinition) {
try {
//根据BeanDefinition的getBeanClassName方法获取class信息并且存入map容器
Class<?> beanClass = Class.forName(registry.getBeanDefinition(beanDefinitionName).getBeanClassName());
beanClassAndObjectMap.put(beanClass, beanClass.newInstance());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException e) {
e.printStackTrace();
}
} }
} /* 遍历存放Class和实例化完的对象,并进行获取属性注解的处理操作 */
private void executeSetField() {
beanClassAndObjectMap.keySet().forEach(this::doSetHasAnnotationsFiled);
} /* 开始对包含自定义注解的属性进行赋值 */
private void doSetHasAnnotationsFiled(Class beanClass) {
this.currentClass = beanClass; try {
//通过反射技术,获取对象的所有属性对象
Field[] fields = beanClass.getFields(); for (Field field : fields) {
// 获取每个属性上所有被装饰的注解
Annotation[] annotations = field.getAnnotations(); // 遍历这些注解对象,用来判断是否有我自定义的注解
for (Annotation annotation : annotations) {
// 执行注解MyValue处理
doMyValueHandler(field, annotation);
// 执行注解MyValue2处理
doMyValue2Handler(field, annotation);
}
}
} catch (Exception e) {
e.printStackTrace();
}
} /* 处理@MyValue注解 */
private void doMyValueHandler(Field field, Annotation annotation) throws Exception {
if (annotation.annotationType() == MyValue.class) {
doHandler(annotation, "value", field);
}
} /* 处理@MyValue2注解 */
private void doMyValue2Handler(Field field, Annotation annotation) throws Exception {
if (annotation.annotationType() == MyValue2.class) {
doHandler(annotation, "size", field);
}
} // 真正的解析注解属性内容地方
// 获取注解属性值并对对象的属性进行赋值操作
private void doHandler(Annotation annotation, String value, Field field) throws Exception {
// 通过注解获取执行处理对象类
InvocationHandler invocationHandler = Proxy.getInvocationHandler(annotation);
// 获取注解的属性信息列表(Map形式)
Field memberValues = invocationHandler.getClass().getDeclaredField("memberValues");
// 设置属性访问权限
memberValues.setAccessible(true);
// 通过属性信息列表获取我的注解属性值信息
Map map = (Map) memberValues.get(invocationHandler);
// 给对象的属性进行动态赋值操作 (关键点)
field.set(this.beanClassAndObjectMap.get(currentClass), map.get(value));
// 梳理完注解赋值后的bean对象后,存到set集合中,后期用于把对象存到beanFactory的缓存中
// 后期getBean()方法获取到的就是我们自定义填充完属性的对象
hasMyAnnotationObjects.add(currentClass);
} /* 遍历填充完的对象属性,注册到BeanFactory中 */
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
this.hasMyAnnotationObjects.forEach(dataClass -> {
beanFactory.registerSingleton(pareBeanName(dataClass), beanClassAndObjectMap.get(dataClass));
});
} /* 首字母小写 */
private String pareBeanName(Class dataClass) {
return dataClass.getSimpleName().toLowerCase().charAt(0)
+ dataClass.getSimpleName().substring(1);
}
}

6.启动Spring

这个时候就只剩下启动类了:

public static void main(String[] args) {
ClassPathXmlApplicationContext app =
new ClassPathXmlApplicationContext("spring.xml");

7.执行效果

可以查看到在BeanFactory中定义的BeanDefinition信息





通过代理类对注解进行参数解析





可以查看到通过注解赋值成功后的对象





执行结果

总结:

本文使用Spring其中一个扩展点BeanDefinitionFactoryPostProcessor接口,和注解的技术进行结合。再通过反射的机制查看哪些类的哪些属性是我们需要去处理的。处理后对象放到统一的容器内,后期再注册到spring的工厂中。整体思路就是这样,如有不理解地方请大家指出 一起进步~

Java注解如何对属性动态赋值的更多相关文章

  1. SpEL + AOP实现注解的动态赋值

    一.自定义注解 先聊聊这个需求,我需要根据用户的权限对数据进行一些处理,但是痛点在哪里呢?用户的权限是在请求的时候知道的,我怎么把用户的权限传递给处理规则呢?想了以下几种方案: Mybatis 拦截器 ...

  2. java自定义注解知识实例及SSH框架下,拦截器中无法获得java注解属性值的问题

    一.java自定义注解相关知识 注解这东西是java语言本身就带有的功能特点,于struts,hibernate,spring这三个框架无关.使用得当特别方便.基于注解的xml文件配置方式也受到人们的 ...

  3. java 注解,动态代理

    秒懂,Java 注解 (Annotation)你可以这样学 深入理解Java注解类型(@Annotation) 注解可以理解为标签. 当开发者使用了Annotation 修饰了类.方法.Field 等 ...

  4. C# 实体/集合差异比较,比较两个实体或集合值是否一样,将实体2的值动态赋值给实体1(名称一样的属性进行赋值)

    /// <summary> /// 实体差异比较器 /// </summary> /// <param name="source">源版本实体& ...

  5. 深入JAVA注解之属性注解

    项目目录结构 实体类: package org.guangsoft.annotation.entity; import java.lang.annotation.ElementType; import ...

  6. JAVA 注解教程(三)注解的属性

    简介 注解的属性也叫做成员变量,注解只有成员变量,没有方法.注解的成员变量在注解的定义中以“无形参的方法”形式来声明,其方法名定义了该成员变量的名字,其返回值定义了该成员变量的类型 实例 @Targe ...

  7. Java初学者作业——完成对已定义类(Admin)的对象的创建。并完成属性的赋值和方法的调用。

    返回本章节 返回作业目录 需求说明: 完成对已定义类(Admin)的对象的创建.并完成属性的赋值和方法的调用. 实现思路: 创建 MyTest 类,并添加 main函数. 在 main函数中完成对 A ...

  8. 秒懂,Java 注解 (Annotation)你可以这样学 - CSDN博客

    https://blog.csdn.net/briblue/article/details/73824058 文章开头先引入一处图片. 这处图片引自老罗的博客.为了避免不必要的麻烦,首先声明我个人比较 ...

  9. JAVA 注解的基本原理

    以前,『XML』是各大框架的青睐者,它以松耦合的方式完成了框架中几乎所有的配置,但是随着项目越来越庞大,『XML』的内容也越来越复杂,维护成本变高. 于是就有人提出来一种标记式高耦合的配置方式,『注解 ...

随机推荐

  1. 【spring源码系列】之【Bean的实例化】

    人生需要探索的热情.坚持的勇气以及热爱生活热爱自己的力量. 1. Bean的实例化 上一篇讲述了bean的生命周期,其中第一步就涉及到了bean的实例化,本文重点分析bean实例化,先进入源码中的Ab ...

  2. 18、linux文件属性

    文件的描述信息: [root@centos6 /]# ls -lih 总用量 118K 3538945 drwxr-xr-x 3 root root 4.0K 8月 23 17:12 app 3276 ...

  3. layui laydate 设置日期格式 最大值等

    laydate.render({ elem: "#jhsj", format: 'yyyy-MM', type: 'month', //显示月份 year 显示到年 max : & ...

  4. 使用Dice loss实现清晰的边界检测

    ​ 前言: 在深度学习和计算机视觉中,人们正在努力提取特征,为各种视觉任务输出有意义的表示.在一些任务中,我们只关注对象的几何形状,而不管颜色.纹理和照明等.这就是边界检测的作用所在. 关注公众号CV ...

  5. Docker搭建Prometheus+grafana监控系统

    一.Prometheus简介 1.简介 Prometheus是由SoundCloud开发的开源监控报警系统和时序列数据库(TSDB). Prometheus使用Go语言开发,是Google BorgM ...

  6. Swoole异步投递task任务

    [使用场景] Swoole的task模块可以用来做一些异步的慢速任务.耗时场景.如webim中发广播,发送邮件等,把这些任务丢给task进程之后,worker进程可以继续处理新的数据请求,任务完成后会 ...

  7. shell运维习题训练

    注:初学shell,以下为本人自己写的答案,如果有更好的,请指教! 1. 求2个数之和: 2. 计算1-100的和 3. 将一目录下所有的文件的扩展名改为bak 4.编译并执行当前目录下的所有.c文件 ...

  8. java面试一日一题:字节java后端工程师面试题

    今天来分享下字节一面面试题,各位小伙伴看看都能答上来吗,弄懂下面的问题你离字节又近了一步哦,加油吧 1.自我介绍: 2.问到项目中为什么选择hbase,如果有多个查询条件如何设置数据存储方案: 3.t ...

  9. java封装继承以及多态(含代码)

    封装 该露的露,该藏的藏 我们常需设计要追求,"高内聚,低耦合".高内聚就是类的内部数据操作细节自己完成.不允许外部干涉:低耦合:仅暴漏少量的方法给外部使用. 封装(数据的隐藏) ...

  10. FirstDay

    昨天心血来潮,想着注册一博客,没想到今天再登时,审阅就通过了,多少有点庆辛.从今天起,我也算是有博客的人了! 为什么选博客园开通?好多IT论坛里都允许有博文,CSDN感觉过于高大上,其他系列论坛大多内 ...