Java的反射

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集合的,会报类型转换错误

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值