这里是想介绍一下如何通过Java的注解机制,实现对bean资源的注入。主要介绍实现的方法,至于例子的实用性不必讨论。
需求:一个应用有两个数据库,分别为DB-A,DB-B。
假设持久层框架使用iBatis来实现,那么SqlMapClient对象在创建时,对于两个不同的DB连接要有两个不同的SqlMapClient对象,
假设我们有一个Service类为MyService.java,该类中有两个SqlMapClient对象sqlMapA、sqlMapB分别对应着DB-A、DB-B。
先看看我们的SqlMapClient.java类:(自定义SqlMapClient类,用来演示。)
需要使用到的技术:
1、使用@interface:实现对需要进行注入的对象使用注解进行标注
2、配合Java反射:实现对注解进行标注的,类、属性、方法,通过反射进行代理的操作权限处理
3、注解处理类:注解需要处理的业务操作,从而使注解在标注过程存在真实的意义
实现过程:
1、自定注解类:
package com.wangyong.learnproject.annotation;
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 DataSource {
/*
* dao 的类型
*/
String type() default "A";
/*
* sql-map-config文件的路径,用于加载iBatis的sqlMapClient对象
*/
String sqlMap() default "";
/*
* 协议方式
*/
DataSource.Protocol protocol() default DataSource.Protocol.DEFAULT;
public static enum Protocol {
DEFAULT,
DUBBO,
REST;
private Protocol() {
}
}
}
2、自定义注解处理类的接口:(提供注解需要做的业务操作接口)
package com.wangyong.learnproject.annotation;
/*
* 封装类业务处理说明:
* 自定义注解需要进行注解业务处理的接口
* 定义资源注入的接口IFieldWiring
* 之所以这里要定义这个接口,是为了以后拓展用,我们很方便的定义更多的自定义注解
*/
import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
public interface IFieldWiring {
Class<? extends Annotation> annotationClass();
void wiring(Object object, Field field);
}
3、自定义注解处理类的业务处理服务类:(提供注解需要做的业务操作实际业务处理)
package com.wangyong.learnproject.annotation;
//import com.mchange.v2.reflect.ReflectUtils;
import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
/*
* 封装类业务处理说明:
* 自定义注解进行业务处理的服务方法类
* 通过实现自定义注解处理的接口的方法使其对外提供业务处理
*/
public class DataSourceWiring implements IFieldWiring {
@Override
public Class<? extends Annotation> annotationClass() {
return DataSource.class;
}
@Override
public void wiring(Object object, Field field) {
Object fileObj = ReflectUtils.getFieldValue(object,field.getName());
if (fileObj != null){
return;
}
DataSource annotation = field.getAnnotation(DataSource.class);
String type = annotation.type();
String sqlMap = annotation.sqlMap();
//这里可以用缓存来实现,不用每次都去创建新的sqlMapClient对象
SqlMapClient sqlMapImpl = new SqlMapClient(sqlMap,type);
//将生成的sqlMapClient注入到bean对象的字段上
ReflectUtils.setFieldValue(object,field.getName(),SqlMapClient.class,sqlMapImpl);
}
}
4、Java反射操作工具:
package com.wangyong.learnproject.annotation;
import org.apache.commons.lang.StringUtils;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class ReflectUtils {
/*
* 取得字段值
*/
public static Object getFieldValue(Object object, String fieldName){
if (object == null || fieldName == null || "".equals(fieldName)){
return null;
}
Class<?> clazz = object.getClass();
try{
String methodname = "get"+ StringUtils.capitalize(fieldName);
Method method = clazz.getDeclaredMethod(methodname);
method.setAccessible(true);
return method.invoke(object);
}catch (Exception e){
try {
Field field = clazz.getDeclaredField(fieldName);
field.setAccessible(true);
return field.get(object);
}catch (Exception e1){
e1.printStackTrace();
}
}
return null;
}
public static void setFieldValue(Object target,String fname,Class<?> fieldClass,Object fieldObj){
if (!fieldClass.isAssignableFrom(fieldObj.getClass())){
return;
}
Class<?> clazz = target.getClass();
try{
Method method = clazz.getDeclaredMethod("set"+Character.toUpperCase(fname.charAt(0))+fname.substring(1),fieldClass);
method.setAccessible(true);
method.invoke(target,fieldObj);
}catch (Exception e){
try {
Field field = clazz.getDeclaredField(fname);
field.setAccessible(true);
field.set(target,fieldObj);
}catch (Exception e1){
e1.printStackTrace();
}
}
}
}
5、调用注解处理业务方法使注解生效
package com.wangyong.learnproject.annotation;
/*
* 封装类业务处理说明:
* 已经基本大功告成,只要将我们的DataSourceWiring.java类使用起来即可
* MyAnnotationBeanProcessor.java,这个类主要用于为bean对象注入资源
*/
import java.lang.reflect.Field;
import java.util.List;
public class MyAnnotationBeanProcessor {
/**
* 函数业务处理含义
* 注入资源
*
* @param serviceObject
* @param fieldAutoWirings 所有实现IFieldWiring的接口的对象,我们可以在此扩展
* @return
*/
public void wire(Object serviceObject, List<IFieldWiring> fieldAutoWirings) throws Exception{
Class<?> cls =serviceObject.getClass();
for (Field field:cls.getDeclaredFields()){
for (IFieldWiring fieldAutoWiring:fieldAutoWirings){
if (field.isAnnotationPresent(fieldAutoWiring.annotationClass())){
fieldAutoWiring.wiring(serviceObject,field);
break;
}
}
}
}
}
6、需要使用注解进行注入(bean)的实体类
package com.wangyong.learnproject.annotation;
/*
* 封装类业务处理说明:
* 自定义sqlMapClient类,用来演示
* 模拟数据操作的客户端bean
* 需要注入的bean,通过自定义的注解进行注入
*/
import org.apache.commons.lang.builder.ToStringBuilder;
import org.apache.commons.lang.builder.ToStringStyle;
import java.util.Map;
//@SupperssWarnings("unchecked")
public class SqlMapClient {
private String type = null;
private String sqlMap = null;
public SqlMapClient(String type, String sqlMap) {
this.type = type;
this.sqlMap = sqlMap;
}
public String selectForObject(String sql, Map in){
return this.toString();
}
@Override
public String toString() {
return new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE).append("sqlMap",sqlMap).append("type",type).toString();
}
}
7、进行bean资源注入引用的服务类
package com.wangyong.learnproject.annotation;
/*
* 封装类业务处理说明:
* 自定义sqlMapClient类,用来演示
* 模拟数据操作的客户端bean
* 需要注入的bean,通过自定义的注解进行注入
*/
import org.apache.commons.lang.builder.ToStringBuilder;
import org.apache.commons.lang.builder.ToStringStyle;
import java.util.Map;
//@SupperssWarnings("unchecked")
public class SqlMapClient {
private String type = null;
private String sqlMap = null;
public SqlMapClient(String type, String sqlMap) {
this.type = type;
this.sqlMap = sqlMap;
}
public String selectForObject(String sql, Map in){
return this.toString();
}
@Override
public String toString() {
return new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE).append("sqlMap",sqlMap).append("type",type).toString();
}
}
8、启动测试类:
package com.wangyong.learnproject.annotation;
import java.util.ArrayList;
import java.util.List;
public class FieldWiringTest {
public static void main(String[] args) throws Exception{
MyAnnotationBeanProcessor processor = new MyAnnotationBeanProcessor();
MyService b = new MyService();
List iFieldWiring = new ArrayList();
iFieldWiring.add(new DataSourceWiring());
processor.wire(b,iFieldWiring);
System.out.println(b.selectForObjectFromA("",null));
System.out.println(b.selectForObjectFromB("",null));
}
}
运行结果:
SqlMapClient[sqlMap=A,type=com/annotation/sql-map-config-a.xml]
SqlMapClient[sqlMap=B,type=com/annotation/sql-map-config-B.xml]
Process finished with exit code 0
由执行结果可以说明DataSource资源已经被我们正确的注入了。
如果想扩展的话,只需要新建一个类实现IFieldWiring接口即可。假设叫InParamWiring.java,实现了接口定义的两个方法后,在使用的时候,只要用以下代码便可将资源注入了: