黑马程序员技术交流社区

标题: 【石家庄校区】Java反射 [打印本页]

作者: xiekai_sjz    时间: 2019-7-13 21:44
标题: 【石家庄校区】Java反射
    反射是Java的一个特点,也是使原本为静态语言的Java,多了那么一些灵活性,在理解各个框架源码以及组件内容的时候是一个不错的知识点,比如注解,这是一个非常常见,又很好使的玩意,之前也有简单的学习---Java 注解 基础、Java 注解 实践
从主要以下几点开始学习
Class类 和 面向对象

在面向对象的环境中,万事万物皆对象,但也总有例外,Java中有两个不属于对象,一个是普通数据类型,一个是静态的成员
普通数据类型有封装类的弥补,静态的属于类,那么类是不是对象呢,类是对象,是java.lang.Class类的实例对象,看文字还比较容易理解,中文说出来就比较绕口, 英文: there is a class named Class
一个普通的类的实例对象表示

[AppleScript] 纯文本查看 复制代码
public class Coo {
    //Doo的实例对象 以doo表示
    Doo doo=new Doo();
}

class Doo{}
那么一个Class类的实例对象,有三种表示方式,但不能是new Class,因为下面源码中也解释为什么
[Java] 纯文本查看 复制代码
/*
* Private constructor. Only the Java Virtual Machine creates Class objects.
* This constructor is not used and prevents the default constructor being
* generated.
*/
private Class(ClassLoader loader) {
    // Initialize final field for classLoader.  The initialization value of non-null
    // prevents future JIT optimizations from assuming this final field is null.
    classLoader = loader;
}
上面是Class类中的一个构造器,是私有的,而且注释说只有JVM创建Class对象,在以前的Java版本中你可能会看到一个无参的构造器,没关系,那你的构造器也绝对是私有,不能直接创建,在上面中出现了一个JIT编译的关键词,有兴趣的小伙伴可以研究扩展
任何类都是Class的实例对象,下面三种方式:
[Java] 纯文本查看 复制代码
public class Coo {

    //Doo的实例对象 以doo表示
    Doo doo=new Doo();

    //①可以看出Doo类有一个隐含的静态成员变量class
    Class first=Doo.class;

    //②已知类的对象,通过getClass获取
    Class second=doo.getClass();

    //重理解一次:doo代表Doo类的实例对象,first、second代表的是Class的实例对象
    //这个Class的实例对象又证明说Doo这个类本身是一个实例对象的存在
    //一本正经的胡说八道,那么给一个官方给出的说法是这样的 :first、second表示了Doo类的类 类型(class type)
    //所有东西都是对象,类也是对象,是Class的实例对象,这个对象称为该类的类类型
    //就可以分析到,Doo的对象是doo,Doo的类类型是Class的对象first、second
    //不管哪种表达方式表示Doo的类类型,一个类只可能是Class类的一个实例对象,所以first == second

    //③需要异常处理,参数为类的全称"com.cloud.eureka.Doo"
    Class third=null;

    {
        try {
            third = Class.forName("com.cloud.eureka.Doo");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }

    //依然存在 first == second == third
    //由此可见,我们可以通过类的类类型创建该类的对象,通过first、second、third创建
    //需要异常处理,是谁的类的类类型对象,创建的对象就是谁,需要强转
    //newInstance前提需要无参构造方法
    {
        try {
            Doo dooFirst= (Doo) first.newInstance();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }
}

class Doo{}
Java 动态加载类信息
三种表示Class的实例对象中,第三种具有很好的动态加载类③
很多时候,大家都是通过工具(IDEA、eclipse等)进行办公或者学习,编译和运行都是由工具来辅助完成的,那么我们需要知道编译、运行的区别
1.2..3...好,我们得到了编译、运行知识的技能
只要是在类里面用到的,都隐含class,对应的类的类类型,如下:
[Java] 纯文本查看 复制代码
public class Coo {

    Class c0=int.class;
    Class c1=String.class;
    Class c2=Double.class;
    Class c3=void.class;
   
    // package不是在类里面的,error
    // Class c4=package.class;
}

在Doo类中,写个方法
[Java] 纯文本查看 复制代码
class Doo{
    public static void staticVoidMethod(Object o){
        //传递的是什么类型,就是什么类型
        Class co=o.getClass();
    }
}
o传递的是什么对象,co就是该类的类类型,那么底层怎么实现的,可能会比较复杂,贴一份源码
public final native Class<?> getClass();
这是一个native声明的一个方法,称为本地方法,Java中有一项技术JNR,使用Java声明,C语言实现,Java 中调用...一堆,有兴趣的可以了解了解,效果就是上面说的,返回类的类类型
下面是简单的通过Class获取类的信息:
[Java] 纯文本查看 复制代码
class Doo {
    public static void staticVoidMethod(Object o) {
        //传递的是什么类型,就是什么类型
        Class co = o.getClass();

        System.out.println("类的全名称:" + co.getName());
        System.out.println("类的名字:" + co.getSimpleName());

        //Method类,方法对象
        //一个成员方法 就是 一个Method对象
        //getMethods 获取所有public的方法,其中包括父类继承的函数
        Method[] allMethods = co.getMethods();

        //getDeclaredMethods获取该类自己声明的方法
        Method[] thisMethods = co.getDeclaredMethods();

        for (Method method : allMethods) {
            //method.getReturnType()得到的是类的类类型
            //比如返回值是String,那么得到的是String.class的类类型,通过getName获取名称
            System.out.println("返回类型:" + method.getReturnType().getName());

            System.out.println("方法名称:" + method.getName());

            //获取参数类型
            Class[] parameterTypes = method.getParameterTypes();
            for (Class c : parameterTypes) {
                System.out.println("参数类型:" + c.getName());
            }
            System.out.println("====================================");
        }

        //成员变量 =》对象
        //属于java.lang.reflect.Field
        //Field封装了关于成员变量的操作
        //getFields获取所有public的成员变量
        Field[] field=co.getFields();
        //得到自己声明的成员变量
        Field[] declaredFields=co.getDeclaredFields();
        for (Field fields:field) {
            System.out.println("成员变量类型"+fields.getType());
            System.out.println("成员变量名称"+fields.getName());
        }
    }
}
简单来一个main方法,加入一个String类
[Java] 纯文本查看 复制代码
public class Coo {
    public static void main(String[] args) {
        String hello=new String();
        Doo.staticVoidMethod(hello);
    }
}
控制台打印 , 所有String内的方法信息:
[Java] 纯文本查看 复制代码
类的全名称:java.lang.String
类的名字:String
返回类型:boolean
方法名称:equals
参数类型:java.lang.Object
====================================
返回类型:java.lang.String
方法名称:toString
====================================
返回类型:int
方法名称:hashCode
====================================
返回类型:int
方法名称:compareTo
参数类型:java.lang.Object
====================================
//......

可以总结出来,getDeclaredXXX()方法都是获取自己声明的内容,包括成员变量,构造器,方法等等,直接的getXXX()方法部分会获取所有内容包括父类的内容,另外数组是一个特殊的存在,打印的是“0]”差不多的样子,在JVM对数组的存储方式也比较VIP,有兴趣的可以理解扩展
方法的反射
上面有获取所有的方法的示例,下面来学习如何获取某一个方法以及方法的反射操作
①方法的名称和方法的参数列表可以唯一定位某一个方法
②method.invoke(对象,参数列表)
[Java] 纯文本查看 复制代码
public class MethodReflect {
    //获取getMethod方法,获取①号
    public static void main(String[] args) {
        MethodDemo demo = new MethodDemo();
        //1.获取类信息
        Class c0 = demo.getClass();

        //2.获取方法
        try {
            //第一种写法
            Method method1 = c0.getDeclaredMethod("getMethod", new Class[]{String.class, String.class});
            //第二种写法
            Method method2 = c0.getDeclaredMethod("getMethod", String.class, String.class);

            //平时正常的调用方法: demo.getMethod(str0,str1)
            //现在使用method1来调用--public Object invoke(Object obj, Object... args)
            //第一个参数是调用的类,第二个参数是可用可无,按定义的方法来录入(str0,str1)
            //invoke的方法如果有返回值,则返回Object的值,void的返回值为null
            try {
                Object object1 = method1.invoke(demo, new Object[]{"hello", " world"});
                Object object2 = method2.invoke(demo, "hello", " world");
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            } catch (InvocationTargetException e) {
                e.printStackTrace();
            }

        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        }

    }

}

class MethodDemo {
    //①
    public void getMethod(String a, String b) {
        System.out.println("concat: " + a + b);
    }

    //②
    public void getMethod(int a, int b) {
        System.out.println("sum: " + a + b);
    }
}

反射和泛型
泛型不说了,非常的常用,比如list,map等等,约定类型,不多做解释,直接先来一个操作,比对List和List<String>是否相等
[Java] 纯文本查看 复制代码
public class Coo {
    public static void main(String[] args) {
        //无泛型
        List list0=new ArrayList();
        //String泛型
        List<String> list1=new ArrayList<>();
        list1.add("hello");

        Class c0=list0.getClass();
        Class c1=list1.getClass();
        //输出
        System.out.println(c0==c1);
    }
}
输出的结果是true,  编译后的class文件也可以当成字节码,说明反射的操作都是编译之后的操作,而且返回true说明编译之后list的泛型被抹去了,去泛型化的,得到Java的泛型是一种规范,只在编译时有效,跳过编译编译就无效了,为了验证这一点,刚好可以使用反射来做一个验证
[Java] 纯文本查看 复制代码
//获取list1<String> 中的 add方法 ,向里面加一个int类型的值
Method method=c1.getMethod("add",Object.class);
method.invoke(list1,100);

System.out.println(list1.size());

list1的大小改变了,说明添加成功了,也验证了Java泛型只在编译期有效,运行时则去泛型化,如果去遍历这个list1是会报类型转化异常的
反射的用处有很多,比如工具类,源码理解,注解解析等等,再例如excel导出导入这样的操作,网上也有非常多的poi操作案例,也可以用反射+注解的方式非常简洁的实现; 例如spring源码中很多的注解@Autowired、@SpringCloudApplication、@Service...等等很多很多



1.png (50.37 KB, 下载次数: 15)

1.png





欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2