Java 程序中,所有的对象都有两种类型:编译时类型和运行时类型,而很多时候对象的编译时类型和运行时类型不一致。 Object obj = new String(“hello”);
obj.getClass()
例如:某些变量或形参的声明类型是 Object 类型,但是程序却需要调用该对象运行时类型的方法,该方法不是 Object 中的方法,那么如何解决呢?
解决这个问题,有两种方案:
Reflection(反射)是被视为动态语言的关键,反射机制允许程序在运行期间借助于 Reflection API 取得任何类的内部信息,并能直接操作任意对象的内部属性及方法。
加载完类之后,在堆内存的方法区中就产生了一个 Class 类型的对象(一个类只有一个 Class 对象),这个对象就包含了完整的类的结构信息。我们可以通过这个对象看到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构,所以,我们形象的称之为:反射。
针对于给写好的.java源文件进行编译(使用javac.exe),会生成一个或多个.class字节码文件。接下来,我们使用java.exe命令对指定的.class文件进行解释运行。这个解释运行的过程中,我们需要将.class字节码文件加载(使用类的加载器)到内存中(存放在方法区)。加载到内存中的.class文件对应的结构即为Class的一个实例。
比如。加载到内存中的Person类或String类或User类,都作为Class的一个一个的实例。
class clazz1 = Person.class;
class claz2z2 = String.class;
class clazz3 = User.class;
class clazz4 = Comparable.class;
说明:运行时类在内存中会缓存起来,在整个执行期间只会加载一次。
@Test
public void test1() throws ClassNotFoundException {
//1.调用运行时类的静态属性class
Class clazz1 = User.class;
System.out.println(clazz1);
//2.调用运行时类的对象的getClass()
User user = new User();
Class clazz2 = user.getClass();
System.out.println(clazz2);
//3.调用Class的静态方法forName(String className)
Class clazz3 = Class.forName("p145.User");
System.out.println(clazz3);
//4.使用类的加载器
Class<?> clazz4 = ClassLoader.getSystemClassLoader().loadClass("p145.User");
System.out.println(clazz4);
}
所以Java都可以使用。
负责类的加载,并对应于一个Class的实例。
引导类加载器/启动类加载器。
扩展类加载器。
系统类加载器/应用程序类加载器。
通过Class的实例调用newInstance()方法即可。
在JDK9中被标识为过时,替换为通过Constructor类调用newInstance(…)。
public class OtherTest {
@Test
public void test1() throws ClassNotFoundException {
//获取运行时类的父类
Class clazz = Class.forName("p129.test1.Person");
Class superClass = clazz.getSuperclass();
System.out.println(superClass);
}
@Test
public void test2() throws ClassNotFoundException {
//获取运行时类实现的接口
Class clazz = Class.forName("p129.test1.Person");
Class[] interfaces = clazz.getInterfaces();
for (Class c :
interfaces) {
System.out.println(c);
}
}
@Test
public void test3() throws ClassNotFoundException {
//获取运行时类所在的包
Class clazz = Class.forName("p129.test1.Person");
Package aPackage = clazz.getPackage();
System.out.println(aPackage);
}
@Test
public void test4() throws ClassNotFoundException {
//获取运行时类带泛型的父类
Class clazz = Class.forName("p129.test1.Person");
Type genericSuperclass = clazz.getGenericSuperclass();
System.out.println(genericSuperclass);
}
@Test
public void test5() throws ClassNotFoundException {
//获取运行时类的父类的泛型
Class clazz = Class.forName("p129.test1.Person");
//获取带泛型的父类(Type是一个接口,Class实现了该接口)
Type superclass = clazz.getGenericSuperclass();
//如果父类是带泛型的,则可以强转为ParameterizedType
ParameterizedType paramType = (ParameterizedType) superclass;
//调用getActualTypeArguments()获取泛型参数,结果是一个数组,因为可能有多个泛型参数
Type[] arguments = paramType.getActualTypeArguments();
//获取泛型参数的名称
System.out.println(((Class)arguments[0]).getName());
}
}
指的是调用指定的属性、方法和构造器。
public class ReflectTest {
@Test
public void test2() throws Exception {
Class clazz = Person.class;
Person per = (Person) clazz.newInstance();
//1.通过Class实例调用getDeclaredField(String fieldName)获取运行时类指定名的属性
Field nameFiled = clazz.getDeclaredField("name");
//2.确保此属性可以访问
nameFiled.setAccessible(true);
//3.通过Field的实例调用get和set方法来获取或设置此属性的值
nameFiled.set(per,"张三");
System.out.println(nameFiled.get(per));
}
}
@Test
public void test3() throws Exception {
Class clazz = Person.class;
Person per = (Person) clazz.newInstance();
//1.通过Class实例调用getDeclaredMethod(String methodName,Class...args)获取指定方法
Method setNameMethod = clazz.getDeclaredMethod("setName", String.class);
//2.确保此方法可以访问
setNameMethod.setAccessible(true);
//3.通过Method的实例调用invoke(Object obj,Object...args),即为对Method对应方法的调用
//invoke方法的返回值即为Method对应方法的返回值
//若Method对应方法的返回值为void,则invoke方法返回值为null
Object returnValue = setNameMethod.invoke(per, "10");
System.out.println(returnValue);
}
@Test
public void test4() throws Exception {
Class clazz = Person.class;
//1.通过Class实例调用getDeclaredConstructor(Class...args)获取指定参数类型的构造器
Constructor constructor = clazz.getDeclaredConstructor(String.class, int.class);
//2.确保此方法可以访问
constructor.setAccessible(true);
//3.通过Constructor的实例调用newInstance(Object...args),返回一个运行时类的实例
Person per = (Person) constructor.newInstance("tom", 12);
System.out.println(per);
}
public class FruitTest {
@Test
public void test1() throws Exception {
//1.读取配置文件中的信息,获取全类名
Properties pros = new Properties();
File file = new File("C:\\Users\\22923\\Desktop\\Java\\Java SE\\src\\p192\\config.properties");
FileInputStream fis = new FileInputStream(file);
pros.load(fis);
String fruitName = pros.getProperty("fruitName");
//2.通过反射创建指定全类名对应的类的实例
Class clazz = Class.forName(fruitName);
Constructor constructor = clazz.getDeclaredConstructor();
constructor.setAccessible(true);
Fruit fruit = (Fruit) constructor.newInstance();
//3.通过榨汁机对象调用run()
fruit.squeeze();
}
}