本篇内容介绍了“java反射的用途及实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!
反射是什么?
反射是Java程序开发语言的特征之一,它允许动态地发现和绑定类、方法、字段,以及所有其他的由于有所产生的的元素。通过反射,能够在需要时完成创建实例、调用方法和访问字段的工作。
反射机制主要提供功能
在运行时判断任意一个对象所属的类
在运行时构造任意一个类的对象
在运行时判断任意一个类所具有的成员变量和方法
在运行时调用任意一个对象的方法,通过反射甚至可以调用到private修饰的方法
生成动态代理
反射在牛逼框架中的使用
Spring 框架的 IOC 基于反射创建对象和设置依赖属性。
Spring MVC 的请求调用对应方法,也是通过反射。
JDBC 的 Class#forName(String className) 方法,也是使用反射。
反射中,Class.forName 和 ClassLoader 区别?
这两者,都可用来对类进行加载。差别在于:
Class#forName(…) 方法,除了将类的 .class 文件加载到JVM 中之外,还会对类进行解释,执行类中的 static 块。
ClassLoader 只干一件事情,就是将 .class 文件加载到 JVM 中,不会执行 static 中的内容,只有在 newInstance 才会去执行 static 块。
反射的常用类
Java中反射相关的类大部分都在rt.jar下java.lang.reflect中,其实需要的类并不多,主要有以下几个:
java.lang.Class
Class类的实例表示正在运行的Java类和接口。
java.lang.reflect.Field
提供有关类或者接口的属性信息,以及对它的动态访问权限。反射的字段可能是一个类(静态)属性或实例属性,简单的理解可以把它看成一个封装反射类的属性的类。有点绕,慢慢体会吧。
java.lang.reflect.Constructor
提供关于类的单个构造方法的信息以及对它的访问权限。这个类和Field类不同,Filed类封装类反射类的属性,而Constructor类则封装类反射类的构造方法。
java.lang.reflect.Method
提供关于类和接口上单个方法的信息。所反映的方法可能是类方法或者实例方法(包括抽象方法)。这个类不难理解,他的作用就是用来封装反射类方法的一个类。
java.lang.reflect.Modifier
提供了用于解码类和成员访问修饰符的静态方法和常量。修饰符集合被表示为具有表示不同修饰符的不同位位置的整数。
java.lang.reflect.Array
提供了动态创解决数组和访问数组的静态方法,该类中的所有方法都是静态方法。
反射的优缺点
优点
可以在程序运行的过程中,操作这些对象。
可以解耦,提高程序的可扩展性。
缺点
因为是JVM操作,所以对于性能来说会有所下降。
容易对程序源码造成一定的混乱。
探索 Class
java文件编译后变成class文件,class文件被类加载器加载到内存中,并且JVM根据其字节数组创建了对应的Class对象。
Class类是Java反射的起源,针对任何一个我们想使用的类,只有先为它产生一个Class对象,接下来就可以通过Class对象获取其他的信息。
JVM为每个类管理着一个独一无二的Class对象,当我们需要创建每个类的对象时,JVM会检查所要加载的类对应的Class对象是否已经存在。不存在,则JVM会根据类加载机制加载并创建对应的Class对象,最后使用Class对象创建出我们通常使用的实例对象。
获取Class类的三种方式
1.调用Object类的getClass()方法获得Class对象。
2.使用Class类的forName("com.tian.XXX")静态方法获取与字符串对应的对象(类或接口的全限定名)。
3.使用.class获取该类性的Class对象。
Class常用方法
方法非常之多。
获取类信息
了解了Java反射的详细细节之后,我们可以使用反射机制来获取类中的信息。
创建对象
使用无参构造方法创建对象
比如说下面这段代码:
Class clazz = Class.forName("java.lang.String"); String str = (String)clazz.newInstance();
这里需要注意,这个类必须是有无参构造方法,不然这种方式会报错的。
使用有参构造方法
可以使用三个步骤来完成:
1.获取指定类对应的Class对象
2.通过Class对象获取满足指定参数类型要求的构造方法类对象
3.调用指定的Constructor对应的newInstance方法,传入对应的参数值,创建出我们想要的实例对象。
Class clazz = Class.forName("java.lang.String"); Constructor constructor = clas.getConstructor(String.class); String str = (String)constructor.newInstance("hello world");
这样就创建了一个String对象实例。
调用方法
前面已经聊过Method这个类,我们可以通过Method类中的invoke方法动态调用器方法。
public final class Method extends Executable { public Object invoke(Object obj, Object... args){ //.... } }
这个方法的第一个参数是一个对象类型,表示要在指定的这个对象上调用这个方法(方法名称)。第二个参数是可变参数,用来给这个方法传递参数值;
invoke方法里返回的值用来表示动态调用指定方法后的返回值。如果调用私有的方法,先调用setAccessible(true)来曲线Java语言堆笨方法的访问检查,然后再调用invoke方法来真正执行这个私有方法。
访问成员变量的值
使用反射可以获取类的成员变量的对象代表,成员变量的对象代表是
java.lang.reflect.Field类的实例,可以使用他的getXyy()方法来获取指定对象上的值,也可以使用setXyy()方法来动态修改指定对象上的值,其中xyy是成员变量。
比如说:setAge(22);其中age就是成员变量。
操作数组
数组也是一个度一项,可以通过反射来查看数组的各个属性的信息,比如
ingt [] intArr=new Int[10]; Sysytem.out.prinlt("数组类型:"+intArr.getClass.getComponentType().getName()); Object obj=Array.newInstance(int.class, 10); //维数组赋值 for(int i=0;i<10;i++){ Array.setInt(obj,i,i); } for(int i = 0;i<10;i++){ System.out.print("第"+i+"好元素为"+Array.getInt(obj,i)); }
反射与动态代理
代理模式是Java中使用频率相当高的设计模式之一,尤其是在牛逼的框架中,Spring,Mybatis,Dubbo等框架中。
其中反射就是一个很好的应用。
静态代理模式我们就没有必要提他了,相当于一个业务需要代理,你就得给他搞一个代理类。全是手动搞出来的。
动态代理的原理就是,在程序运行时候根据需要动态地创建目标类的代理对象,典型应用场景:
JDK动态代理
CGlib动态代理
“java反射的用途及实现”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注亿速云网站,小编将为大家输出更多高质量的实用文章!
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。