Java中的反射机制

xiaoxiao2021-02-28  67

一、什么是反射?

对于程序员来说,应该很少需要直接使用反射工具;之所以在语言中提供它们,是为了支持其他Java特性,比如对象序列化、Java Beans以及RMI。还有就是在很多框架中,也是应用到了反射机制。

在初学时,只知道这是个生成驱动实例的语句,而这其实就是应用了反射

// Load the driver Class.forName( "sun.jdbc.odbc.JdbcOdbcDriver"); Connection c = DriverManager.getConnection( dbUrl, user, password);

二、反射能做什么?

主要的四个功能: - 获取对象所属的类 - 获取类的成员变量、方法 - 运行时创建对象 - 运行时调用对象的方法

通过对象获取包名、类名

实例.getClass().getName()

package Reflection; class Demo { // your code here } public class Test { ; public static void main(String[] args) { Demo demo = new Demo(); System.out.println(demo.getClass().getName()); } } 获取class类

有以下三种方式

package Reflection; class Demo { // your code here } public class Test { ; public static void main(String[] args) { Class<?> demo1 = null; Class<?> demo2; Class<?> demo3; // 1. class.forName("类的路径") try { demo1 = Class.forName("Reflection.Demo"); } catch (ClassNotFoundException e) { e.printStackTrace(); } // 2. 实例.getClass() demo2 = new Demo().getClass(); // 3. 类名.class demo3 = Demo.class; if (demo1 != null) { System.out.println(demo1.getName()); } System.out.println(demo2.getName()); System.out.println(demo3.getName()); } } 运行时创建对象、调用对象的方法 package Reflection; class Dog { private String name; private String owner; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getOwner() { return owner; } public void setOwner(String owner) { this.owner = owner; } @Override public String toString() { return this.name + "'s owner is " + this.owner; } } public class Test { ; public static void main(String[] args) { Class<?> demo = null; try { // 使用反射机制加载类Dog demo = Class.forName("Reflection.Dog"); } catch (ClassNotFoundException e) { e.printStackTrace(); } Dog dog = null; // 在运行时动态地创建类Dog的对象 try { dog = (Dog)demo.newInstance(); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } // 运行时调用对象的方法 if (dog != null) { dog.setName("qiuqiu"); } if (dog != null) { dog.setOwner("lxb"); } System.out.println(dog); } }

还有需要注意一点的是,如果我们没自定义无参构造函数,而是直接定义了有参构造函数,会报以下错误

null java.lang.InstantiationException: Reflection.Dog at java.lang.Class.newInstance(Class.java:427) at Reflection.Test.main(Test.java:49) Caused by: java.lang.NoSuchMethodException: Reflection.Dog.<init>() at java.lang.Class.getConstructor0(Class.java:3082) at java.lang.Class.newInstance(Class.java:412) ... 1 more Process finished with exit code 0

所以在程序运行时创建对象时,我们需要自定义无参构造函数

class Dog { private String name; private String owner; // 自定义的无参构造函数 public Dog() {} // 有参构造函数 public Dog(String name, String owner) { this.name = name; this.owner = owner; } // 省略 } 其它功能

可用构建器创建新对象,通过Constructor类取得其它类的构造函数

用get()和set()方法读取和修改与Field对象关联的字段

用invoke()方法调用与Method对象关联的方法,即调用其他类的方法

此外,我们可调用方法getFields(),getMethods(),getConstructors(),分别返回用于表示字段、方法以及构建器的对象数组,取得并修改数组的信息

取得其他类的父类、实现的接口、权限修饰符、属性类型等

三、反射的优缺点

优点:可以实现运行时动态创建对象,增加程序的灵活性。这种灵活性怎么体现的呢?比如我们在开发一个大型的软件,编译发布后,我们很可能需要更新一些功能,但因为软件已发布,我们肯定不能要求用户把以前的卸载,再重新安装新的版本。如果我们采用静态的解决方法的话,需要重新编译整个程序,而如果采用反射机制的话,就可以不用卸载,只需要在运行时才动态的创建和编译,就可以实现更新的功能。缺点: 性能影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它满足我们的要求。这类操作总是慢于只直接执行相同的操作。破坏封装,因为通过反射可以访问私有变量或方法,这样可能会存在安全性的问题。比如我们知道String是不可变的,因为这个类被final修饰说明不可继承,并且String其实就是被final修饰的一个字符数组value,value是private的,但是通过反射是可以访问到private成员变量,因此通过反射,我们其实是可以改变String的,很明显,这样是不安全的。
转载请注明原文地址: https://www.6miu.com/read-2631504.html

最新回复(0)