通过注解获取和改变Bean的某变量值

news/2024/5/19 4:00:59 标签: java, 反射, Bean值的修改, Java注解, Annotation

Java有时需要通过自定义注解,获取某Bean的某变量的值,根据业务要求处理数据,然后再把新值设置回Bean的同一变量中,下面我们简要介绍一下,如何实现,

1,自定义注解的定义

java">import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

// 注解的作用范围,必用,
// 如用在类上,方法上,属性上等,更多参考ElementType枚举
@Target(ElementType.FIELD)

// 注解的生命周期,即注解的保留时间阶段,必用,
// SOURCE,表示该注解的生命周期只在编译阶段,
// CLASS,该注解被保留在class文件上
// RUNTIME,该注解生命周期在运行时
@Retention(RetentionPolicy.RUNTIME)

// 是一个标记注解,没有定义属性,作用是为了表示该注解可以被继承,非必用
// 使用了该元注解的自定义注解,应用到某父类上,则该类的子类也默认继承了该注解
// @Inherited

// 是一个标记注解,里面没有任何属性,非必用
// 用 @Documented 注解修饰的注解类会被 JavaDoc 工具提取成文档
// @Documented

// 允许在目标对象重复使用该注解, 指定重复类,仅参考暂不用,非必用
// @Repeatable(BeanFieldChecks.class) 
public @interface BeanFieldCheck {
	/** 校验结果信息*/
	String message()'
	
	/** 校验器*/
	Class validator();	
}
java">// 仅提供参考,不使用
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface BeanFieldChecks {
	
	/** 注解复数,即同一变量可以多次使用同一注解, 
	可以不使用,单个注解已可以完成事情 */
	BeanFieldCheck[] values();	
}

2,注解在Bean中的使用

java">public class User {
	@ApiModelProperty(value="主键ID")
	private Long id;
	
	// 注解使用
	@ApiModelProperty(value="用户编码")
	@JavaField(message="编码不能为空", validator=UserValidator01.class)
	private String userCode;
	
	// 注解使用
	@ApiModelProperty(value="用户名称")	
	@JavaField(message="姓名不能含特殊字符和重名", validator=UserValidator02.class)
	private String userName;
	
	@ApiModelProperty(value="年龄")
	private Interger age;
	
	// ...getter/setter省略

}

3,注解的业务处理逻辑

根据注解要求获取Bean某变量的值,按业务要求处理,之后把新值赋给原变量,以 UserValidator02.java 为例子:

java">public class UserValidator02 {

	// 方法功能:根据Bean的变量注解,获取类变量的原值,
	// 再根据注解数据处理,重新赋给原来的变量
	public static <T> T updateModelValue(T bean) {
		if (null == bean) {
			return null;
		}
		// 获取目标model的java类
		Class cls = bean.getClass();
		
		// 获取类的所有字段变量
		Field[] fields = cls.getDeclareFields();
		for (Field field : fields) {
			// 获取某个类型的注解为:
			//Annotation an = field.getAnnotation(BeanFieldCheck.class)
			// 获取字段的全部注解
			Annotation ans = field.getAnnotations();
			if (null == ans || ans.length < 1) {
				continue;
			}
			
			// 遍历X字段上的所有注解
			for (Annotation an : ans) {
				if (null == an) {
					continue;
				}
				
				// 对象的变量名, 不做它用,仅显示用法
				String fieldName = field.getName();
				
				// 获取指定类型注解内message的值
				String tempVal = "";				
				if (an instanceof BeanFieldCheck) {
					// 获取Bean类变量的该类型注解中message配置项值
					BeanFieldCheck anItem = (BeanFieldCheck)an;
					tempVal = anItem.message();		
				}
								
				try {
					// 类变量是private私有,需改变为可访问性
					field.setAccessible(true);
					
					//获取类变量的原来值
					Object oldVal = field.get(bean);
					
					// 这里可根据自己业务要求处理数据,此处仅简单暂时
					if (null != oldVal) {
						tempVal = oldVal.toString() + "," + tempVal;
					}
					
					// 将新值赋回类变量的该字段中
					field.set(bean, tempVal);
					
					// 恢复变量private特性
					field.setAccessible(false);
				} catch(Exception e) {
					e.printStackTrace();
				}		
			}
		}
		//返回修改类变量后的Bean
		return bean;
	}
	
	// 其它方法省略	
}

4,如何触发执行注解的业务逻辑

上面定义了自定义的注解,校验器逻辑,以及注解的使用,但怎么触发没有,
可以通过代码主动调用触发,例如:

java">User newUser = UserValidator02.updateModelValue(user);

还可以更好一点,推荐定义一个AOP切面,让使用了 BeanFieldCheck 注解的地方在切面中自动触发进行校验处理,AOP切面不在本中展开,简要提供如下:

java">@Component	//使该类被springIOC扫描到,交由spring管理
@Aspect 	//定义为aop切面类
public class UserCheckAspect {
	
	// 定义方法作用的目标表达式
	@Pointcut(value="@annotation(com.aa.bb.cc.BeanFieldCheck") 
	public void myPointcut01(){}
	
	// 根据需要定义和修改
	@Pointcut(value = "execution(* com.aa.bb.cc.BeanFieldCheck..*(..))")
	public void myPointcut02(){}

	@Before(value="myPointcut01()")
	public void beforeExecute(JoinPoint joinPoint){
		// 调用UserValidator02.updateModelValue()处理
		//TODO...方法执行前的逻辑处理...
	}

	@Around(value="myPonitcut01()")
	public void aroundExecute(ProceedingJoinPoint joinPoint) throws Throwable{
		// 调用UserValidator02.updateModelValue()处理
		//TODO...方法执行前后的逻辑处理...
	}	

	@After(value="myPonitcut01()")
	public void afterExecute(JoinPoint joinPoint) throws Throwable{
		// 调用UserValidator02.updateModelValue()处理
		//TODO...方法执行后的处理...
	}
}

以上展示了,自定义注解的写法,应用,以及如何通过注解对Java的Bean中某个变量的获取原值,数据处理,赋给原来变量的方法和过程,在实际使用根据自己需要相应的修改,欢迎拍砖讨论...


http://www.niftyadmin.cn/n/322210.html

相关文章

C语言设计扫雷(保姆级教学)

目录 引入&#xff1a; 游戏思路与逻辑&#xff1a; 游戏具体实现过程 整个程序的大体逻辑&#xff1a; 游戏的执行逻辑&#xff1a; game.h头文件内容&#xff1a; 具体函数讲解 menu&#xff08;菜单函数&#xff09; Initboard&#xff08;初始化棋盘函数&#xff09…

实例35---字符串反转,如将字符串 “www.runoob.com“ 反转为 “moc.boonur.www“。

文章目录 前言一、题目描述 二、题目分析 三、解题 1.解题方法一--- for循环来将字符的下标数值进行对每一个字符进行交换 ---程序运行代码 2.解题方法二------ 指针 ------ 程序运行代码(1)程序运行代码优化 总结 前言 本系列为C语言菜鸟100道基础经典例题详解刷题系列。点滴…

什么是Java中的finalize()方法?它有什么作用

在Java中&#xff0c;finalize()方法是一个由Object类定义的方法&#xff0c;用于在对象被垃圾回收器回收之前执行一些清理工作。finalize()方法是一个被保护的方法&#xff0c;可以被子类重写&#xff0c;但是通常情况下不需要显式地调用该方法。 finalize()方法的作用 在Jav…

java中List与AbstractList

一、List 接口 List 接口继承了 Collection 接口&#xff0c;在 Collection 接口的基础上增加了一些方法。相对于 Collection 接口&#xff0c;我们可以很明显的看到&#xff0c;List 中增加了非常多根据下标操作集合的方法&#xff0c;我们可以简单粗暴的分辨一个方法的抽象方…

systemctl针对service类型的配置文件

文章目录 systemctl针对service类型的配置文件systemctl配置文件相关目录简介systemctl配置文件的设置项目简介两个vsftpd运行的实例多重的重复设置方式&#xff1a;以getty为例将tty数量由6个降低为4个暂时新增vsftpd到1212端口 自己做个服务 systemctl针对service类型的配置文…

C++——动态管理(类和对象收尾)

作者&#xff1a;几冬雪来 时间&#xff1a;2023年5月14日 内容&#xff1a;C内存管理讲解 目录 前言&#xff1a; 1.类的对象&#xff08;收尾&#xff09;&#xff1a; 1.友元函数&#xff1a; 2.内部类&#xff1a; 3.匿名对象&#xff1a; 4.优化&#xff1a; 2.…

Hive on Spark调优(大数据技术2)

第2章 Yarn配置 2.1 Yarn配置说明 需要调整的Yarn参数均与CPU、内存等资源有关&#xff0c;核心配置参数如下。 &#xff08;1&#xff09;yarn.nodemanager.resource.memory-mb 该参数的含义是&#xff0c;一个NodeManager节点分配给Container使用的内存。该参数的配置&am…

Mutipart

含义&#xff1a; 多部分的、复合 场景&#xff1a; 位置&#xff1a;package org.springframework.boot.autoconfigure.web.servlet; Springboot中autoconfigration包下web包下servlet下DispatcherServletAutoConfiguration中&#xff1a; 有一个默认加载的Bean Bean …