自学笔记(二) -- 反射 Reflection

反射

前言: 作为Java学习的过程中最绕不开的就是反射了。反射的机制丰富了Java的语言的全面性,但是反射也是我认为Java中最难学的部分。当前Java生态中主流框架的实现都是非常依赖反射的,所以学会反射非常重要。

什么是反射?
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

类加载完之后,在堆内存的方法区就产生一个Class类型的对象(一个类就对应一个Class对象),这个对象包含这个类的完整的结构信息。通过这个对象可以获取这个类的所有方法,方法参数,返回值等很多信息。

反射调用
实例化对象–>getClass()方法–>得到完整“包类”名称

正常调用
引入需要的“包类”名称–>通过new实例化–>取得实例化对象

反射的基本使用

 public class TestReflection {
    public static void main(String[] args) throws ClassNotFoundException {
        //获取反射对象
        Class c1 = Class.forName("com.liu.pojo.User");
        System.out.println(c1);

			//判断同一个类获取多个Class是否获取的是一个对象,通过hashCode比较
			//一个类在内存中只有一个class对象
			//一个类被加载后,类的整个结构都会被封装在class中
        Class c2 = Class.forName("com.liu.pojo.User");
        Class c3 = Class.forName("com.liu.pojo.User");
        Class c4 = Class.forName("com.liu.pojo.User");
        System.out.println(c2.hashCode());
        System.out.println(c3.hashCode());
        System.out.println(c4.hashCode());
    }
}

结果:
反射对象hashCode比较
三种获取Class的方法

//通过类名获得
Class clazz = User.class;
//通过对象获得
Class clazz = person.getClas();
//通过路径获得
Class clazz = Class.forName("demo.Person");
//基本内置类型的包装类
Class clazz = Integer.TYPE;
//获得父类类型
Class superClazz = clazz.getSuperclass();

反射的一些方法

//获取系统类的加载器
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();

//获取系统类加载器的父类加载器 --> 扩展类加载器
ClassLoader parent = systemClassLoader.getParent();

//获取扩展类加载器的父类加载器 --> 跟加载器(C/C++)
ClassLoader parent1 = parent.getParent();

//获取当前类的加载器-->系统类的加载器
ClassLoader parent2 = Class.forName("包路径").getClassLoader();

//JDK内置类的加载器-->根加载器加载
ClassLoader parent3 = Class.forName("java.lang.object").getClassLoader();

//获取系统类加载器可以加载的路径
System.getProperty("java.class.path");

反射获取类的信息

Class c1 = Class.forName("com.liu.pojo.User");

//获取类的名字
c1.getName();
c1.getSimpleName();

//获取类的属性
Field[] publicFields = c1.getFields();//public权限
Field[] fields = c1.getDeclaredFields();//所有
Field name = c1.getDeclaredField("name");//所有

//获取类的方法
Method[] publicMethods = c1.getMethods();//获得本类及其父类的方法
Method[] methods = c1.getDeclaredMethods();
Method getName = c1.getMethod("getName", null);

//获得指定的构造器
Constructor[] declaredConstructors = c1.getDeclaredConstructors();
Constructor[] constructors = c1.getConstructors();

反射创建对象

Class c1 = Class.forName("com.liu.pojo.User");

//构造一个对象
User user = (User) c1.newInstance();//本质调用类的无参构造器

//通过构造器创建对象
Constructor constructor= c1.getDeclaredConstructor();
User user2 = (User) constructor.newInstance(001, "name", 18);

通过反射调用方法及属性

Class c1 = Class.forName("com.liu.pojo.User");

//构造一个对象
User user = (User) c1.newInstance();//本质调用类的无参构造器
//反射调用方法
Method setName = c1.getDeclaredMethod("setName", String.class);
setName.invoke(user, "小刘");

//反射调用属性  不能直接操作私有属性需要关掉安全检测
Field name = c1.getDeclaredField("name");
name.setAccessible(true);//设置权限true-允许
name.set(user, "小小刘");

当程序多次使用反射时建议将 setAccessible() 设为true

反射的操作泛型用来获取反射获取的方法的参数及返回值

Class c1 = Class.forName("com.liu.pojo.User");

Method method = c1.getDeclaredMethod("method", String.class);

//返回方法的参数类型
Type[] genericParameterTypes = method.getGenericParameterTypes();
for (Type genericParameterType : genericParameterTypes) {
    //如果这个参数属于参数化类型 map等
    if (genericParameterType instanceof ParameterizedType) {
        //获取真实参数
        Type[] actualTypeArguments = ((ParameterizedType) genericParameterType).getActualTypeArguments();
    }
}

//返回方法的返回值类型
Type genericReturnType = method.getGenericReturnType();

通过反射操作注解

public class TestAnnotationOfReflection {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
        Class<?> c1 = Class.forName("com.liu.main.User");

        //获取全部注解
        Annotation[] annotations = c1.getAnnotations();
        for (Annotation annotation : annotations) {
            System.out.println(annotation);
        }

        TableUser annotation = c1.getAnnotation(TableUser.class);
        String value = annotation.value();
        System.out.println(value);

        //获得类指定的注解
        Field field = c1.getDeclaredField("name");
        FieldUser annotationOfField = field.getAnnotation(FieldUser.class);
        String s = annotationOfField.columnName();
        String type = annotationOfField.type();
        int length = annotationOfField.length();
    }
}

@TableUser("db_user")
class User {

    @FieldUser(columnName = "db_name", type = "varchar", length = 10)
    String name;
    @FieldUser(columnName = "db_age", type = "int", length = 10)
    int age;

    public User() {
    }
}

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface TableUser {
    String value();
}

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface FieldUser {
    String columnName();

    String type();

    int length();
}

参考链接
B站-西部开源

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值