Java 反射四大核心类:Class、Constructor、Method、 Filed

简介: Java 反射四大核心类:Class、Constructor、Method、 Filed

反射可以做到的事情

  • 在不修改源码的情况下,通过配置文件来控制程序
  • 在程序运行的时候,通过反射动态创建对象,并且可以操作对象的所有属性、方法、构造器
  • 生成动态代理

反射重要的4个类

  • java.lang.Class
  • java.lang.reflect.Field
  • java.lang.reflect.Method
  • java.lang.reflect.Constructor

Class类

一个类在堆中只有一个Class对象,这个Class对象包含了类的完整结构信息

在使用反射的过程中,把方法、属性、构造器都当做对象来处理的

Class类对象由系统创建,继承了Object类

Class类的类图

17892935686842cb859eb5747cc090bb.png

Class类常用方法

//1.通过全路径 获取对应的Class对象
String classAllPath = "com.java.test.Student";
Class<?> cls = Class.forName(classAllPath);
//2.获取包名
String packageName = cls.getPackage().getName();
//3.获取全类名
String classFullName = cls.getName();
//4.通过Class对象创建对应对象的实例
Object obj = cls.newInstance();
//5.获取public修饰的属性,并设置值
Field nameField = cls.getField("name");
String nameValue = nameField.get(obj).toString();
nameField.set(obj,"jack");
System.out.println(nameField.get(obj)); //jack
//6.获取所有public修饰的属性(字段)
Field[] fields = cls.getFields();
for (Field field : fields) {
    System.out.println(field.getName());
}
//7.获取所有属性,包括私有属性
Field[] declaredFields = cls.getDeclaredFields();
for (Field declaredField : declaredFields) {
    System.out.println(declaredField.getName());
}

获取Class类对象4种方式

  • Class.forName( )
  • 类名.class
  • 对象名.getClass( )
  • 类加载器获取Class对象
//1.通过Class类的静态方法forName()获取
String classAllPath = "com.java.test.Student";
Class<?> cls = Class.forName(classAllPath);
//2.类名.class
Class<String> stringClass = String.class;
//3.对象名.getClass():通过已经创建好的对象,获取Class对象
String str = new String("hello");
Class<? extends String> aClass = str.getClass();
//4.通过类加载器来获取类的Class对象
ClassLoader classLoader = str.getClass().getClassLoader();
//ClassLoader classLoader = String.class.getClassLoader();
Class<?> strClassObject = classLoader.loadClass("java.lang.String");

基本数据类型的Class类对象和包装类的Class类对象

//byte short int long float double char boolean
Class<Byte> byteClass = byte.class;
Class<Short> shortClass = short.class;
Class<Integer> integerClass = int.class;
Class<Long> longClass = long.class;
Class<Float> floatClass = float.class;
Class<Double> doubleClass = double.class;
Class<Character> characterClass = char.class;
Class<Boolean> booleanClass = boolean.class;
Class<Integer> type = Integer.TYPE;
//其它包装类获取Class类对象的方式和Integer一样

Field类

  • getField( )、getFields( )、getDeclaredFields( )
  • getModifiers( ):返回修饰符
  • getType( ):返回类型
  • getName( ):返回属性名
//通过全路径 获取对应的Class对象
String classAllPath = "com.java.test.Student";
Class<?> cls = Class.forName(classAllPath);
//获取所有属性对象,包括私有属性
Field[] declaredFields = cls.getDeclaredFields();
for (Field declaredField : declaredFields) {
    System.out.println(declaredField.getModifiers()+" "+
            declaredField.getType()+" "+declaredField.getName());
}

Method类

  • getMethod( )、getMethods( )、getDeclaredMethods( )
  • getModifiers( ):返回方法的返回值类型
  • getParameterTypes( ):返回参数类型的数组
  • getName( ):返回方法的名称
//通过全路径 获取对应的Class对象
String classAllPath = "com.java.test.Student";
Class<?> cls = Class.forName(classAllPath);
获取所有方法对象,包括私有方法
Method[] declaredMethods = cls.getDeclaredMethods();
for (Method declaredMethod : declaredMethods) {
    System.out.println(declaredMethod.getModifiers()+" "+declaredMethod.getReturnType()+" "+
            Arrays.toString(declaredMethod.getParameterTypes()));
}

Constructor类

  • getConstructor( )、getConstructors( )、getDeclaredConstructors( )
  • getModifiers( ):以int形式返回修饰符
  • getName( ):返回构造器的全类名
  • getParameterTypes( ):返回参数类型的数组
  • setAccessible( ):爆破,访问私有成员
  • newInstance( ):创建对象
//通过全路径 获取对应的Class对象
String classAllPath = "com.java.test.Student";
Class<?> cls = Class.forName(classAllPath);
//获取所有的构造器对象,并且获取构造器对象的信息
Constructor<?>[] constructors = cls.getDeclaredConstructors();
for (Constructor<?> constructor : constructors) {
    System.out.println(constructor.getModifiers()+" "+constructor.getName()
            +" "+Arrays.toString(constructor.getParameterTypes()));
}
// 使用反射创建对象
// 无参构造器创建对象
Object o = cls.newInstance();
// 有参构造器创建对象
Constructor<?> constructor = cls.getConstructor(int.class, String.class);
Object o2 = constructor.newInstance(1000, "hello");
// 私有有参构造器创建对象
Constructor<?> declaredConstructor = cls.getDeclaredConstructor(String.class);
declaredConstructor.setAccessible(true);
Object o3 = declaredConstructor.newInstance("jack");

反射的简单使用

/** 需求:
   1.根据配置文件的信息创建对象
   2.通过反射操作private修饰的属性的值,并调用对象的方法
*/

propeties配置文件

classFullPath=com.java.reflection.Student
name=jack
method=sayHello

Student类

@Data
public class Student {
    private String name;
    public void sayHello(){
        System.out.println("hello "+name);
    }
}

main( )方法

public static void main(String[] args) throws Exception {
    //1.使用Properties类读取Maven工程resources文件夹下的配置文件信息
    //1.1读取配置文件到内存中
    Properties properties = new Properties();
    InputStream resourceAsStream = DemoTest.class.getResourceAsStream("/test.properties");
    properties.load(resourceAsStream);
    //1.2获取配置文件信息
    String classFullPath = properties.get("classFullPath").toString();
    String nameProValue = properties.get("name").toString();
    String methodPro = properties.get("method").toString();
    //2.使用反射机制创建对象,并调用对象的方法和属性
    Class<?> cls = Class.forName(classFullPath); //获取运行时类对象
    Object obj = cls.newInstance();//根据类对象创建对应的实例对象
    Method methodObj = cls.getMethod(methodPro);//获取方法的对象
    //获取属性的对象,并设置属性值
    Field fieldObj = cls.getDeclaredField("name");
    fieldObj.setAccessible(true);//私有属性必须先爆破,才能设置值
    fieldObj.set(obj,nameProValue);
    //通过方法对象调用方法
    methodObj.invoke(obj);//hello jack
}



目录
相关文章
|
26天前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
44 8
|
25天前
|
监控 Java
Java基础——反射
本文介绍了Java反射机制的基本概念和使用方法,包括`Class`类的使用、动态加载类、获取方法和成员变量信息、方法反射操作、以及通过反射了解集合泛型的本质。同时,文章还探讨了动态代理的概念及其应用,通过实例展示了如何利用动态代理实现面向切面编程(AOP),例如为方法执行添加性能监控。
|
26天前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
59 1
|
1月前
|
存储 安全 Java
java.util的Collections类
Collections 类位于 java.util 包下,提供了许多有用的对象和方法,来简化java中集合的创建、处理和多线程管理。掌握此类将非常有助于提升开发效率和维护代码的简洁性,同时对于程序的稳定性和安全性有大有帮助。
63 17
|
1月前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
114 4
|
1月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
58 2
|
1月前
|
存储 安全 Java
如何保证 Java 类文件的安全性?
Java类文件的安全性可以通过多种方式保障,如使用数字签名验证类文件的完整性和来源,利用安全管理器和安全策略限制类文件的权限,以及通过加密技术保护类文件在传输过程中的安全。
51 4
|
1月前
|
Java
Java的反射
Java的反射。
26 2
|
1月前
|
Java 编译器 Maven
Java“class file contains wrong class”解决
当Java程序运行时出现“class file contains wrong class”错误,通常是因为类文件与预期的类名不匹配。解决方法包括:1. 确保类名和文件名一致;2. 清理并重新编译项目;3. 检查包声明是否正确。
59 3