博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java的反射
阅读量:4047 次
发布时间:2019-05-25

本文共 4924 字,大约阅读时间需要 16 分钟。

Java的反射

  反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。反射是框架底层的一些方法,比如以后看到的Spring的重要特性DI:控制反转就是这么一个原理,至于为什么使用反射,1.首先你能通过任意对象获取类类型即是所有的信息,这个作用以后才能体现;2.方法的反射的好处就是解耦,比如说a,b,c对象都要调用 print()方法,正常的想法就是要创建每个对象,并且a.print() b.print() c.print() ,但是使用反射的话,就 print()方法的对象.invoke(a,参数列表)想要用哪个对象就用哪个对象

1.1Class类的使用

1)在面向对象的世界里,万事万物皆对象。
   类是对象,类是java.lang.Class类的实例对象。
   There is a class named Class
2)//Foo的实例对象如何表示
     Foo foo1=new Foo();//foo1就表示出来了
   //Foo这个类也是一个实例对象,Class类的实例对象,如何表示呢?
   //任何一个类都是Class的实例对象,这个实例对象有三种表示方式
   //第一种表示方式--->实际在告诉我们任何一个类都有一个隐含的静态成员变量class
     Class c1=Foo.class; 
   //第二种表达方式--->已经知道该类的对象通过getClassF方法
     Class c2=foo1.getClass();
   //官网c1,c2表示了Foo类的类类型(class type),万事万物皆对象,类也是对象,是Class类的实例对象
   //这个对象我们称为该类的类类型
   //不管c1 or c2都代表了Foo类的类类型,一个类只可能是Class类的一个实例对象
   //第三种表达方式
     Class c3=null;
     c3=Class.forName("com.imooc.reflect.Foo");
   //我们完全可以通过类的类类型创建类的对象实例--->通过c1 or c2 or c3创建Foo的实例对象
     Foo foo=(Foo)c1.newInstance();//需要有无参数的构造方法

2.1动态加载类

java编译时刻加载类是静态加载类,运行时刻加载类是动态加载类

new 创建对象是静态加载类,在编译时刻就需要加载所有的可能使用到的类
通过Class a=Class.forName(arg[0]);此时为动态加载,因为编译时不知道使用哪个类,因此编译没有加载任何类,通过编译。运行时,根据 Javac office.java word  (word为arg[0],也是类类型),去确定a是哪个类。这就是动态加载。如果word不存在,此时运行会报错。这就是为何有时候会出现编译通过,运行报错的原因。
动态加载一个好处,就是可以随时增加需要编译的类。例如没有excel类,只有word类,也可以运行,需要excel类时再由程序员写此类(为了能统一控制,word类、excel类需要继承同一个父类或者继承同一个接口)。

   可以理解为装修房子(编译)买电器,我之前计划的有电视冰箱洗衣机,全都随着装修安装完毕了(静态加载),之后呢(运行),觉得少个微波炉,可是根本就没有规划微波炉的线路,那么就需要拆掉原先的线路为新的电器设置安装.而现在我不想那么麻烦,就在装修的时候预留好电源插座这种接口(interface),只要所有的电器都实现了这个接口,有了两孔或者三孔的插头,那么我后期想增添任何电器都可以随时增加(动态加载)

3.1Java获取信息的方法

只要在类里面声明的都有类类型

基本数据类型,void关键字都存在类类型

class.getMethods()方法获取是该类的所有public方法,包括从父类继承的方法;
class.getDeclareMethods()方法获取该类自行声明的所有方法,不论访问权限;
method.getName()获取方法名
method.getReturnType()获取方法的返回值
method.getParameterTypes(),获取方法的参数类型的类类型数组class[]
三获取方法信息:
一、基本的数据类型,void关键字等都存在类类型
Class c = 基类.class (int,String,double,void等)
二、Class类的基本API操作的
1、c.getName()可以获取类的名称
2、c.getSimpleName();//不包含包名的类的名称
3、c.getMethods()获取类的【public方法】集合,【包括继承来的】
***注意【所有方法都是Method类的对象】
4、c.getDeclaredMethods()获取的是所有该类【自己声明】的方法,【不问访问权限】
三、Method类提供了一些操作方法的方法
1、.getReturnType()得到该方法的返回值类型的类类型(class),如int.class String.class
2、.getName()得到方法的名称
3、.getParameterTypes()获得参数列表类型的类类型,如参数为(int,int)则得到(int.class ,int class)
Class c1 = int.class; int的类类型
Class c2 = String.class; String类的类类型 String类字节码
Class c3 = double.class; double这个数据类类型的字节码表示方式
Class c4 = Double.class; Double这个类的类类型字节码表示方式
Class c5 = void.class; 表达了void这个类的类类型
getName为这个类的类类型的具体名称 
c1.getName ---> int
c2.getName ---> java.lang.String 类的全称
c2.getSimpleName ---> String 不包含包名的类的名称

样例如下:

public static void pringClassMessage(Object object){ //要获取类的信息,首先要获取类的类型Class c=object.getClass();//传递的是哪个子类的对象,c就是该子类的类类型//获取类的名称System.out.println("类的名称是:"+c.getName()); /* * Method类,方法对象* 一个成员方法就是一个Method对象* getMethods()方法获取的是所有public函数,包括父类继承而来的* getDeclaredMethods()获取的是所有该类自己声明的方法,不问访问权限* */Method[] ms=c.getMethods();//c.getDeclaredMethods(); for (int i = 0; i < ms.length; i++) { //得到方法的返回值类型的类类型Class returnType=ms[i].getReturnType(); System.out.println(returnType.getName()); //得到方法名System.out.println(ms[i].getName()+"("); //获取参数类型-->得到的是参数列表的类型的类类型Class[] paramType=ms[i].getParameterTypes(); for (Class class1: paramType) { System.out.println(class1.getName()+","); } System.out.println(")"); }  } }

4.1Java获取成员变量构造函数信息

一、成员变量是java.lang.reflect.Field的对象
1、Field类封装了关于成员变量的操作
2、Field[] fs = c.getFields()方法获取所有public的成员变量Field[]信息
3、c.getDeclaredFields获取的是该类自己声明的成员变量信息
4、field.getType()获得成员类型的类类型
5、field.getName()获得成员的名称
二、构造函数是java.lang.Constructor类的对象
1、通过Class.getConstructor()获得Constructor[]所有公有构造方法信息
2、建议getDeclaredConstructors()获取自己声明的构造方法
3、Constructor.getName():String
4、Constructor.getParameterTypes():Class[]
成员变量也是对象,是java.lang.reflect.Field的对象;

5-1 Java 方法反射的基本操作

方法的反射:

1.获取A类中的print(int,int)方法:
 ①要获取一个方法就是获取类的信息,获取类的信息首先要获取类的类类型
  A a1=new A();  Class c= a1.getClass();
 ②获取方法 由名称和参数列表来决定,getMethod获取的是public方法,getDelcaredMethod获取自己声明的方法
  Method m =c.getMethod(methodName,paramtypes);//paramtypes可以用数组的形式  表示new Class[]{int.class,int.class},也可以直接列举类类型
  2.方法的反射操作:是用m对象来进行方法调用,和a1.print(10,20)调用的方法相同 m.invoke(a1,new Object[]{10,20})
Object o=m.invoke(对象名,参数);//方法如果没有返回值返回null,如果有返回值返回具体值,参数可用数组的方式表示,也可以直接列举,没有参数就不写

public Class A{ public void print(){}; public void Print(Sting a,String b){} public void Print(int a,int b){};} public Class B{ public static void main(String[] args){  A a1 = new A();  Class c= a1.getclass;  Method getMet=c.getMethod("print",String.class,String.class);  Object obj=getMet.invoke(a1,"df","df");}}

6-1 Java 通过反射了解集合泛型的本质

1:反射的操作都是编译之后的操作;就是运行阶段
2:java中集合的泛型是防止错误输入的;只在编译阶段有效,只要绕过编译就无效啦
我们可以通过方法的反射来操作,绕过编译
eg:
ArrayList list1=new ArrayList();
ArrayList<String> list2=new ArrayList<String>();
Class c1=list1.getClass();
Class c2=list2.getClass();
System.out.print(c1==c2);//true
Method m=c2.getMethod("add",Object.class);
m.invoke(list2,20);//向list2集合中添加一个int 型的值;绕过编译
当然是不能直接foreach list2集合的,会报类型转换错误

转载地址:http://dofci.baihongyu.com/

你可能感兴趣的文章
妈妈十分生气的shooow
查看>>
怎么写一个温泉管理系统
查看>>
令人神清气爽的shooow
查看>>
指导教师的shooow
查看>>
leetcode面试题01.06.字符串压缩,超出时间限制,样例通过31/32
查看>>
机器学习实战、第二章KNN算法详解、AttributeError: ‘dict‘ object has no attribute ‘iteritems‘
查看>>
leetcode 535 TinyURL 的加密与解密 暴力 年轻人不讲武德—shooter7的博客
查看>>
课程设计(毕业设计)—基于机器学习KNN算法手写数字识别系统—计算机专业课程设计(毕业设计)
查看>>
leetcode1792第232场周赛第三题,以及二维数组根据某一列进行排序——优先队列
查看>>
学生网上选课管理系统的设计与实现—计算机类专业课程设计(毕业设计)
查看>>
新建动态web工程项目红叉报错,以及Could not publish server configuration for Tomcat v9.0 Server at localhost.
查看>>
机器学习SVM的车牌识别系统—计算机专业课程设计(毕业设计)
查看>>
leetcode 80. 删除有序数组中的重复项 II
查看>>
课程设计(毕业设计)—学生宿舍管理系统—计算机类专业
查看>>
毕业设计(课程设计)—SpringBoot网上订餐系统的设计与实现—计算机类专业课程设计(毕业设计)
查看>>
毕业设计(课程设计)—个人博客系统(微博)的设计与实现—计算机类专业课程设计(毕业设计)
查看>>
牛客(中兴捧月)—B-切绳子
查看>>
剑指Offer 13.机器人的运动范围——DFS和BFS
查看>>
Java中GUI编程总结—AWT中的Frame容器、panel面板、布局管理
查看>>
剑指offer12.矩阵中的路径—DFS+剪枝
查看>>