文章目录
提示:以下是本篇文章正文内容,下面案例可供参考
一、关于Java.lang.Class类的理解
1.类的加载过程
程序经过javac.exe命令以后,会生成一个或多个字节码文件(.class结尾)。
接着我们使用java.exe命今对某个字节码文件进行解释运行。相当于某个字节码文件加载到内存中。此过程就称为类的加载。加载到内存中的类,我们就称为运行时类,此运行时类,就作为Class的一个实例。
2.Class的实例就对应着一个运行时类。
3.加载到内存中的运行时类,会缓存一定的时间。在此时间之内,我们可以通过不同的方式来获取此运行时类。
二、获取Class类的实例的四种方式
1 .代码
代码如下(示例):
@Test
public void test3() throws ClassNotFoundException {
//方式一:调用运行时类的属性:class
Class clazz1 = Person.class;
System.out.println(clazz1);
//方式二:通过运行时类的对象
Person p1 = new Person();
Class clazz2 = p1.getClass();
System.out.println(clazz2);
//方式三:调用Class的静态方法:forName(String classPath)
Class clazz3 = Class.forName("com.tyust.edu.Person");
System.out.println(clazz3);
System.out.println(clazz1 == clazz2);
System.out.println(clazz1 == clazz3);
//方式四:使用类的加载器
ClassLoader classLoader = ReflectionTest.class.getClassLoader();
Class clazz4 = classLoader.loadClass("com.tyust.edu.Person");
System.out.println(clazz4);
System.out.println(clazz1 == clazz4);
}
三、Class实例可以是哪些结构的说明:
代码如下(示例):
@Test
public void test4(){
Class c1 = Object.class;
Class c2 = Comparable.class;
Class c3 = String[].class;
Class c4 = int[][].class;
Class c5 = ElementType.class;
Class c6 = Override.class;
Class c7 = int.class;
Class c8 = void.class;
Class c9 = Class.class;
int [] a = new int[10];
int [] b = new int[100];
Class c10 = a.getClass();
Class c11 = b.getClass();
//只要数组的元素类型与维度一样,就是同一个Class
System.out.println(c10 == c11);
}
四、了解类的加载器
五、通过反射创建对应的运行时类的对象
package com.tyust.edu;
import org.junit.Test;
import java.util.Random;
/**
* 通过反射创建对应的运行时类的对象
* @author YML TYUST-XDU 2019-2026
* @create 2023-10-08 10:51
*/
public class NewInstance {
@Test
public void test1() throws InstantiationException, IllegalAccessException {
Class clazz = Person.class;
// newInstance:调用此方法,创建对应的运行时类的对象
//内部调用了运行时类的空参的构造器
//要想此方法正常的创建运行时类的对象,要求:
// 1.运行时类必须提供空参的构造器
//2.空参的构造器的访问权限得够,通常,设置为public
//在javabean中要求提供一个public的空参构造器。原因:
// 1.便于通过反射,创建运行时类的对象
// 2.便于子类继承此运行时类时,默认调用super()时,保证父类有此构造器
Object obj = clazz.newInstance();
System.out.println(obj);
}
//体会反射的动态性
@Test
public void test2(){
int num = new Random().nextInt(3);//0,1,2
String classPath = "";
switch (num){
case 0:
classPath = "java.util.Date";
break;
case 1:
classPath = "java.lang.Object";
break;
case 2:
classPath = "com.tyust.edu.Person";
break;
}
try {
Object obj = getInstance(classPath);
System.out.println(obj);
} catch (Exception e) {
e.printStackTrace();
}
}
public Object getInstance(String classPath) throws Exception{
Class clazz = Class.forName(classPath);
return clazz.newInstance();
}
}
六、获取运行时类的指定结构
1.如何操作运行时类中的指定的属性
2.如何操作运行时类中的指定的方法
3.如何调用运行时类中的指定的构造器
package com.tyust.edu1;
import com.tyust.edu.Person;
import org.junit.Test;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
/**
* 获取运行时类的指定结构
* @author YML TYUST-XDU 2019-2026
* @create 2023-10-08 11:18
*/
public class ReflectionTest {
@Test
public void testField() throws Exception{
Class clazz = Person.class;
//创建运行时类的对象
Person p = (Person) clazz.newInstance();
//获取指定的属性:要求运行时类属性声明为public
//通常不采用此方法
Field id = clazz.getField("id");
//设置当前属性的值
//set():参数1:指明设置哪个对象的属性 参数2:将此属性值设置为多少
id.set(p,1001);
//获取当前属性的值
// get():参数1:获取哪个对象的当前属性值
int pId = (int ) id.get(p);
System.out.println(pId);
}
//如何操作运行时类中的指定的属性
@Test
public void testField1() throws Exception {
Class clazz = Person.class;
//创建运行时类的对象
Person p = (Person) clazz.newInstance();
//1.getDeclaredField(String fieldName) :获取运行时类中指定变量名的属性
Field name = clazz.getDeclaredField("name");
//2.保证当前属性是可访问的
name.setAccessible(true);
//3.获取、设置指定对象的此属性值
name.set(p,"yml");
System.out.println(name.get(p));
}
//如何操作运行时类中的指定的方法
@Test
public void testMethod() throws Exception {
Class clazz = Person.class;
//创建运行时类的对象
Person p = (Person) clazz.newInstance();
//1.获取指定的某个方法
//getDeclaredMethod():参数1:指明获取的方法的名称
//参数2 :指明获取的方法的形参列表
Method show = clazz.getDeclaredMethod("showNation",String.class);
//2.保证当前方法是可访问的
show.setAccessible(true);
//3.
//invoke()的返回值即为对应类中调用的方法的返回值
//参数1:方法的调用者
//参数2:给方法形参赋值的实参
Object returnVaule = show.invoke(p,"CHN");
System.out.println(returnVaule);
System.out.println("*****************如何调用静态方法*********************");
Method showDesc = clazz.getDeclaredMethod("showDesc");
showDesc.setAccessible(true);
//如果调用的运行时类中的方法没有返回值,则此invoke()返回null
Object returnVal = showDesc.invoke(Person.class);
System.out.println(returnVal);//null
}
//如何调用运行时类中的指定的构造器
@Test
public void testConstructor() throws Exception{
Class clazz = Person.class;
//private Person(String name)
//1.获取指定的构造器
//getDeclaredConstructor():参数:指明构造器的参数列表
Constructor constructor = clazz.getDeclaredConstructor(String.class);
//2.保证此构造器是可访问的
constructor.setAccessible(true);
//3.调用此构造器创建运行时类的对象
Person per = (Person)constructor.newInstance("yml");
System.out.println(per);
}
}