java反射

一般情况下,我们使用某个类时必定知道它是什么类,是用来做什么的。于是我们直接对这个类进行实例化,之后使用这个类对象进行操作。

Student student = new student();
student.setage(18)

上面是常见的进行类对象的初始化方式,但我要是一开始并不知道我要初始化的类对象是什么,那该怎么处理呢?

Class clz = Class.forName("com.entity.Student");
Method method = clz.getMethod("setAge", int.class);
Constructor constructor = clz.getConstructor();
Object object = constructor.newInstance();
method.invoke(object, 18);

上面两段代码的执行结果,其实是完全一样的。但是其思路完全不一样,第一段代码在未运行时就已经确定了要运行的类(Student),而第二段代码则是在运行时通过字符串值才得知要运行的类(com.entity.Student)。

所以说什么是反射?

反射就是在运行时才知道要操作的类是什么,并且可以在运行时获取类的完整构造,并调用对应的方法。

一般情况下我们使用反射新建一个对象的步骤包括:

1.获取类的 Class 对象实例:

Class clz = Class.forName("com.entity.Student");

2.根据 Class 对象实例获取 Constructor 对象:

Constructor studentConstructor = clz.getConstructor();

3.使用 Constructor 对象的 newInstance 方法新建反射类对象:

Object studentObj = studentConstructor.newInstance();

如果要调用某一个方法,则需要经过下面的步骤:

1.获取方法的 Method 对象:

Method setAgeMethod = clz.getMethod("setAge", int.class);

2.利用 invoke 方法调用方法:

setAge.invoke(studentObj, 18);

反射常用API----获取反射中的Class对象

1.使用 Class.forName(路径) 静态方法。当你知道该类的全路径名时,你可以使用该方法获取 Class 类对象。

Class clz = Class.forName("com.entity.Student");

2.使用 类名.class 方法。
这种方法只适合在编译前就知道操作的 Class。(那我为啥不直接初始化?)

Class clz = Student.class;

3.使用类对象的 类.getClass() 方法。

Student student = new student();
Class clz = student.getClass()

反射常用API----通过反射创建类对象

1.通过 Class 对象的 newInstance() 方法。

Class clz = Student.class;
Student student = (Student)clz.newInstance();

2.通过 Constructor 对象的 newInstance() 方法。

Class clz = Student.class;
Constructor constructor = clz.getConstructor();
Student student = (Student)constructor.newInstance();

通过 Constructor 对象创建类对象可以选择特定构造方法,而通过 Class 对象则只能使用默认的无参数构造方法。

Class clz = Student.class;
Constructor constructor = clz.getConstructor(String.class, int.class);
//直接给初始化对象属性赋值
Student student = (Apple)constructor.newInstance("张三", 15);

反射常用API----通过反射获取类属性、方法、构造器

1.Class 对象的 getFields() 方法可以获取 Class 类的属性,但无法获取私有属性。

Class clz = Student.class;
Field[] fields = clz.getFields();
for (Field field : fields) {
    System.out.println(field.getName());
}


//输出结果
age

2.使用 Class 对象的 getDeclaredFields() 方法则可以获取包括私有属性在内的所有属性。

Class clz = Student.class;
Field[] fields = clz.getDeclaredFields();
for (Field field : fields) {
    System.out.println(field.getName());
}

//输出结果
name
age

与获取类属性一样,当我们去获取类方法、类构造器时,如果要获取私有方法或私有构造器,则必须使用有 declared 关键字的方法。

Class clz = Student.class;
Method method = clz.getMethod("setAge", int.class);
Method Declaredmethod = clz.getDeclaredMethod("setName", string.class);

实例

我们经常使用的 Spring 中的相关 Bean 的配置就是利用反射去加载对应的类。但是我们要讲的实例不是这个。

Spring 中有一种重要的概念叫做AOP,是在不改原有代码的前提下对其进行增强。而 公共字段自动填充 也是个比较常见的场景。比如上面的例子,假如我们不仅有老师类,又有学生类,它们都有公共字段 姓名、年龄,此时就要用反射的思想实现功能;或者,有的系统需要记录数据创建时间和上一次修改时间,此时也可以用反射的方式确定类的方法。

代码实例:

/**
 * 自定义注解,用于标识某个方法需要进行功能字段自动填充处理
 */
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface AutoFill {
    //数据库操作类型:UPDATE INSERT
    OperationType value();
}

@Aspect
@Component
@Slf4j
public class AutoFillAspect {

    /**
     * 切入点
     */
    @Pointcut("execution(* com.sky.mapper.*.*(..)) && @annotation(com.sky.annotation.AutoFill)")
    public void autoFillPointCut(){}

    /**
     * 前置通知,在通知中进行公共字段的赋值
     */
    @Before("autoFillPointCut()")
    public void autoFill(JoinPoint joinPoint){
        log.info("开始进行公共字段自动填充...");

        //获取到当前被拦截的方法上的数据库操作类型
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();//方法签名对象
        AutoFill autoFill = signature.getMethod().getAnnotation(AutoFill.class);//获得方法上的注解对象
        OperationType operationType = autoFill.value();//获得数据库操作类型

        //获取到当前被拦截的方法的参数--实体对象
        Object[] args = joinPoint.getArgs();
        if(args == null || args.length == 0){
            return;
        }
		
		//获取实体类对象
        Object entity = args[0];

        //准备赋值的数据
        LocalDateTime now = LocalDateTime.now();
        Long currentId = BaseContext.getCurrentId();

        //根据当前不同的操作类型,为对应的属性通过反射来赋值
        if(operationType == OperationType.INSERT){
            //为4个公共字段赋值
            try {
            	//利用反射调用方法
                Method setCreateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_TIME, LocalDateTime.class);
                Method setCreateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_USER, Long.class);
                Method setUpdateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class);
                Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, Long.class);

                //通过调用方法为对象属性赋值
                setCreateTime.invoke(entity,now);
                setCreateUser.invoke(entity,currentId);
                setUpdateTime.invoke(entity,now);
                setUpdateUser.invoke(entity,currentId);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }else if(operationType == OperationType.UPDATE){
            //为2个公共字段赋值
            try {
                //利用反射调用方法
                Method setUpdateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class);
                Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, Long.class);

                //通过调用方法为对象属性赋值
                setUpdateTime.invoke(entity,now);
                setUpdateUser.invoke(entity,currentId);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

@AutoFill(value = OperationType.INSERT)
void insert(Category category);

@AutoFill(value = OperationType.UPDATE)
void update(Category category);

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值