利用反射获得类信息,以获得泛型为例

java源文件编译成.class字节码。运行时加载类的时候就是加载.class文件。这也是java为什么可以一次编译多次运行!

只要我们得到.class文件那就可以获得类信息!可以利用反射!

方式一:对象.getClass() 方法是 根对象Object的方法。 是其他类继承Object的getClass方法。

方式二:类名.class,你可以理解为字节码本身就是静态的,类加载的时字节码就进JVM了。所以类.class好比类调用静态方法似得调用字节码对象。
方式三:Class.forName()是Class类的静态方法。参数是字符串,字符串是类的全路径名。


下面我以反射的方式获得类中的注解为例

我自己测试的文结构,这个例子仅涉及两个文件。


User.java  为model添加注解,然后利用反射获取注解信息。那我们就可以明白ORM框架注解的原理了,原来是利用反射获取到注解的信息,然后取出来进行sql语句拼写。


操作的文件:fanshe.java

package test;
import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import annontation.Colum;
import model.User;
public class fanshe {
        public static void main(String arge[]) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException{
        Class c=Class.forName("model.User");
              User s=(User) c.newInstance();
        //构造对象
        Constructor so=c.getConstructor(int.class,String.class);
        User user=(User) so.newInstance(10,"adb");
       
       
        //获得所有属性
        Field[] fields=c.getDeclaredFields();
        for(Field f:fields){
        Annotation an=f.getDeclaredAnnotation(Colum.class);
        print(an+"");
        }
        Method me=c.getDeclaredMethod("setId",int.class);
        me.invoke(s, 200);
        print(s.getId()+"");
        }
        public static void  print(String s){
        System.out.println(s);
        }
}

结果:

@annontation.Colum(name=table_id, length=10)
@annontation.Colum(name=table_name, length=200)
200

利用反射我们可以做到在运行时改变类的结构信息。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
泛型为Integer的情况下,可以利用反射机制来进行以下操作: 1. 获取泛型型(Type): 可以使用`Class<?> clazz = Integer.class`或者`Class<?> clazz = obj.getClass()`得到泛型型,其中`obj`为已经实例化的泛型对象。 2. 获取泛型型参数(Type Parameter): 可以使用`ParameterizedType parameterizedType = (ParameterizedType) clazz.getGenericSuperclass()`来获取泛型型参数。然后通过`Type[] typeArgs = parameterizedType.getActualTypeArguments()`来获取实际的型参数数组。 3. 创建泛型对象的实例: 可以使用`Integer integer = (Integer) clazz.newInstance()`来创建泛型对象的实例。 4. 调用泛型对象的方法: 可以使用`Method method = clazz.getMethod("methodName", parameterTypes)`来获取泛型中的指定方法,其中`methodName`为方法名,`parameterTypes`为方法的参数型。然后使用`Object result = method.invoke(obj, args)`来调用该方法,其中`obj`为已经实例化的泛型对象,`args`为方法的参数。 5. 使用泛型中的字段: 可以使用`Field field = clazz.getField("fieldName")`来获取泛型中的指定字段,其中`fieldName`为字段名。然后使用`Object value = field.get(obj)`来获取该字段的值,`field.set(obj, value)`来设置该字段的值。 通过反射机制,我们可以动态地获取泛型信息并实例化对象、调用方法以及访问字段。这种灵活性可以在编写通用代码时更方便地操作泛型。但需要注意的是,反射的使用会带来额外的性能开销,因此在性能敏感的场景下需要谨慎使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值