Java反射机制

简介: 什么是反射机制,反射机制有什么作用,反射机制的具体实现

1 什么是反射机制

想要了解反射机制,需要先了解两个概念:编译期和运行期。

编译期:把源代码翻译成机器能识别的代码,比如编译器把java代码编译成jvm识别的字节码文件

运行期:将可执行文件交给操作系统去执行

反射机制:在运行期,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

(简述:java中,只要给定类的名字,就可以在运行时通过反射机制来获得类的所有信息)

2 反射机制的作用

  • 运行时判断对象所属的类
  • 运行时构造一个类的对象
  • 运行时,判断一个类所具有的的成员变量和方法
  • 运行时调用任意一个对象的方法

使用场景

1.反编译(.class变成.java)
2.开发框架,反射是框架设计的灵魂。比如Spring,为了保证框架的通用性,框架需要根据配置文件加载不同的类或对象(运行期间,动态加载所需对象)。

优缺点

对象的创建,分为静态和动态两种。静态:编译时即确定具体类型,绑定对象。动态:运行时才会确定具体类型,能够有效降低类之间的耦合度,最大限度发挥了java的灵活性。

反射机制优点:实现对象的动态创建和编译,体现了很大的灵活性。比如:一个大型软件,不可能一次就设计的很完美,当需要增加新功能时,如何解决?卸载以后重新安装新版本?这是不合适的。静态编译则必须这样做,而反射机制的动态特性可以帮助我们解决这个问题,无需卸载,只需要运行时动态的加载,即可创建和编译新对象。

缺点:对系统性能会产生影响。

3 反射机制的实现

 java中,使用一个类需要把该类加载到虚拟机中,并生成一个Class对象,这个对象保存了该类的所有信息。反射机制的实现,就是获取这个Class对象。

Class对象的获取

第一种:(已存在对象,无需反射)

Student stu1 = new Student();//这一new 产生一个Student对象,一个Class对象。
Class stuClass = stu1.getClass();//获取Class对象
System.out.println(stuClass.getName());

第二种:(需要导入类包,依赖太强)

Class stuClass2 = Student.class;
//判断第一种方式获取的Class对象和第二种方式获取的是否是同一个
System.out.println(stuClass == stuClass2);

第三种:(常用,传入一个字符串即可,或者写在配置文件中)

//注意此字符串必须是真实路径,就是带包名的类路径,包名.类名
Class stuClass3 = Class.forName("cn.bdqn.Student");
//判断三种方式是否获取的是同一个Class对象
System.out.println(stuClass3 == stuClass2);

反射机制常用的类

Java.lang.Class; //类
Java.lang.reflect.Constructor; //构造方法
Java.lang.reflect.Field; //属性
Java.lang.reflect.Method; //方法

4 反射机制的使用

4.1 创建对象

使用反射机制创建对象有两种方式:newInstance()方法和Constructor对象

1)使用Class对象的newInstance()方法创建对象

Class<?> c = Class.forName("cn.bdqn.Student");
Object str = c.newInstance();

2)调用Constructor对象

/获取String的Class对象
Class<?> c = Class.forName("cn.bdqn.Student");
//通过Class对象获取指定的Constructor构造器对象
Constructor constructor=c.getConstructor(String.class);
//根据构造器创建实例:
Object obj = constructor.newInstance(“hello reflection”);

4.2 获取构造方法

获取构造方法

//所有"公有的"构造方法
public Constructor[] getConstructors()
//获取所有的构造方法(包括私有、受保护、默认、公有)
public Constructor[] getDeclaredConstructors()

//获取单个的"公有的"构造方法
public Constructor getConstructor(Class... parameterTypes)
//获取"某个构造方法"可以是私有的,或受保护、默认、公有
public Constructor getDeclaredConstructor(Class... parameterTypes)
调用构造方法

//对象获取构造的方法getConstructor
newInstance(Object... initargs)

示例:

//1.加载Class对象
Class clazz = Class.forName("cn.bdqn.Student");

//2. 获取构造方法
******获取所有公有构造方法******
System.out.println("***所有公有构造方***");
Constructor[] conArray = clazz.getConstructors();
for(Constructor c : conArray){
    System.out.println(c);
}

******获取所有构造方法******
System.out.println("***所有的构造方法(包括:私有、受保护、默认、公有)***");
conArray = clazz.getDeclaredConstructors();
for(Constructor c : conArray){
  System.out.println(c);
}

****** 获取公有、无参的构造方法******
System.out.println("***获取公有、无参的构造方法***");
Constructor constructor = clazz.getConstructor(null);
1>、因为是无参的构造方法所以类型是一个null,不写也可以:这里需要的是一个参数的类    型,切记是类型
2>、返回的是描述这个无参构造函数的类对象。
System.out.println("con = " + con);

3.对象创建
******调用无参构造创建对象****** 
Object newInstance = constructor.newInstance();

******调用有参构造创建对象******
Constructor constructor = clazz.getConstructor(String.class);//参数用以指定类型
constructor.newInstance(new Object[]{"xm"});//参数用以赋值

4.3 获取成员变量并调用

获取成员变量

//批量获取
******获取所有的"公有字段******
Field[] getFields()
******获取所有字段,包括:私有、受保护、默认、公有******
Field[] getDeclaredFields()

//单个获取
******获取某个"公有的"字段******
public Field getField(String fieldName)
******获取某个字段(可以是私有的)******
public Field getDeclaredField(String fieldName)

示例

Class clazz = Class.forName("cn.bdqn.Student");
Object obj = clazz.getConstructor().newInstance();

System.out.println("************获取所有公有的字段********************");
Field[] fieldArray = clazz.getFields();

System.out.println("************获取所有的字段(包括私有、受保护、默认的)********************");
fieldArray = clazz.getDeclaredFields();

System.out.println("******获取公有字段并调用******");
Field f = clazz.getField("name");
//为字段设置值,为Student对象中的name属性赋值--》stu.name = "刘德华"
f.set(obj, "刘德华");
//验证
Student stu = (Student)obj;
System.out.println("验证姓名:" + stu.name);

System.out.println("******获取私有字段并调用******");
f = clazz.getDeclaredField("phoneNum");
System.out.println(f);
f.setAccessible(true);//暴力反射,解除私有限定
f.set(obj, "18888889999");
System.out.println("验证电话:" + stu);

4.4 获取成员方法并调用

获取成员方法

//批量的
public Method[] getMethods():获取所有"公有方法";(包含了父类的方法也包含Object类)
public Method[] getDeclaredMethods():获取所有的成员方法,包括私有的(不包括继承的)

//获取单个
public Method getMethod(String name,Class<?>... parameterTypes):
public Method getDeclaredMethod(String name,Class<?>... parameterTypes)
参数说明:  
  * name : 方法名;
  * Class ... : 形参的Class类型对象
调用方法
public Object invoke(Object obj,Object... args):
参数说明:    

  * obj : 要调用方法的对象;
  * args:调用方式时所传递的实参;
示例

Class clazz = Class.forName("cn.bdqn.Student");
Object obj = clazz.getConstructor().newInstance();

//2.获取所有公有方法
System.out.println("***************获取所有的”公有“方法*******************"); 
Method[] methodArray = clazz.getMethods();

System.out.println("***************获取所有的方法,包括私有的*******************");
Method[] methodArray = clazz.getDeclaredMethods();
System.out.println("***************获取公有的show1()方法*******************");
Method m = stuClass.getMethod("show1", String.class); 
//实例化一个Student对象
Object obj = stuClass.getConstructor().newInstance();
m.invoke(obj, "刘德华");

System.out.println("***************获取私有的show4()方法******************");
m = stuClass.getDeclaredMethod("show4", int.class); 
m.setAccessible(true);//解除私有限定
Object result = m.invoke(obj, 20);
System.out.println("返回值:" + result);    

4.5 通过反射机制运行配置内容

配置文件pro.txt

className=cn.bdqn.Student
methodName=show

测试类

public class Demo {
    public static void main(String[] args) throws Exception {
        //通过反射获取Class对象
        Class stuClass = Class.forName(getValue("className"));//cn.bdqn.Student
        //2获取show()方法
        Method m = stuClass.getMethod(getValue("methodName"));//show
        //3.调用show()方法
        m.invoke(stuClass.getConstructor().newInstance());
   
    }

//此方法接收一个key,在配置文件中获取相应的value
public static String getValue(String key) throws IOException{
    Properties pro = new Properties();//获取配置文件的对象
    FileReader in = new FileReader("pro.txt");//获取输入流
    pro.load(in);//将流加载到配置文件对象中
    in.close();
    return pro.getProperty(key);//返回根据key获取的value值
}

}

相关文章
|
6月前
|
设计模式 人工智能 安全
AQS:Java 中悲观锁的底层实现机制
AQS(AbstractQueuedSynchronizer)是Java并发包中实现同步组件的基础工具,支持锁(如ReentrantLock、ReadWriteLock)和线程同步工具类(如CountDownLatch、Semaphore)等。Doug Lea设计AQS旨在抽象基础同步操作,简化同步组件构建。 使用AQS需实现`tryAcquire(int arg)`和`tryRelease(int arg)`方法以获取和释放资源,共享模式还需实现`tryAcquireShared(int arg)`和`tryReleaseShared(int arg)`。
355 32
AQS:Java 中悲观锁的底层实现机制
|
4月前
|
人工智能 缓存 安全
Java中的反射机制:深入探索与应用
Java反射机制是程序运行时动态获取类信息并操作类成员的特性,具备高度灵活性,但也伴随性能与安全风险。本文详解反射的基本用法、高级应用及最佳实践,助你掌握这一强大工具的正确使用方式。
155 0
|
6月前
|
人工智能 Java 关系型数据库
Java——SPI机制详解
SPI(Service Provider Interface)是JDK内置的服务提供发现机制,主要用于框架扩展和组件替换。通过在`META-INF/services/`目录下定义接口实现类文件,Java程序可利用`ServiceLoader`动态加载服务实现。SPI核心思想是解耦,允许不同厂商为同一接口提供多种实现,如`java.sql.Driver`的MySQL与PostgreSQL实现。然而,SPI存在缺陷:需遍历所有实现并实例化,可能造成资源浪费;获取实现类方式不够灵活;多线程使用时存在安全问题。尽管如此,SPI仍是Java生态系统中实现插件化和模块化设计的重要工具。
222 0
|
4月前
|
人工智能 前端开发 安全
Java开发不可不知的秘密:类加载器实现机制
类加载器是Java中负责动态加载类到JVM的组件,理解其工作原理对开发复杂应用至关重要。本文详解类加载过程、双亲委派模型及常见类加载器,并介绍自定义类加载器的实现与应用场景。
235 4
|
4月前
|
人工智能 安全 Java
掌握Java反射:在项目中高效应用反射机制
Java反射是一种强大功能,允许程序在运行时动态获取类信息、创建对象、调用方法和访问字段,提升程序灵活性。它在框架开发、动态代理、注解处理等场景中广泛应用,如Spring和Hibernate。但反射也存在性能开销、安全风险和代码复杂性,应谨慎使用。
116 0
|
5月前
|
人工智能 Java
Java中的反射机制:深入探索与应用
本文介绍了Java反射机制的基本概念、用途及其实现方式。反射机制允许程序在运行时动态获取类的属性和方法,并调用它们,适用于处理私有成员或权限受限的情况。文章详细讲解了`Class`类的功能,包括获取类的方法、属性、注解、构造器等信息,以及通过四种方式获取`Class`对象的示例代码。此外,还探讨了类加载器、继承关系判断、动态代理等高级内容,展示了如何在运行时创建接口实例并处理方法调用。文末提供了完整的代码示例以加深理解。
120 0
Java中的反射机制:深入探索与应用
|
6月前
|
Java 区块链 网络架构
酷阿鲸森林农场:Java 区块链系统中的 P2P 区块同步与节点自动加入机制
本文介绍了基于 Java 的去中心化区块链电商系统设计与实现,重点探讨了 P2P 网络在酷阿鲸森林农场项目中的应用。通过节点自动发现、区块广播同步及链校验功能,系统实现了无需中心服务器的点对点网络架构。文章详细解析了核心代码逻辑,包括 P2P 服务端监听、客户端广播新区块及节点列表自动获取等环节,并提出了消息签名验证、WebSocket 替代 Socket 等优化方向。该系统不仅适用于农业电商,还可扩展至教育、物流等领域,构建可信数据链条。
|
6月前
|
人工智能 JavaScript Java
Java反射机制及原理
本文介绍了Java反射机制的基本概念、使用方法及其原理。反射在实际项目中比代理更常用,掌握它可以提升编程能力并理解框架设计原理。文章详细讲解了获取Class对象的四种方式:对象.getClass()、类.class、Class.forName()和类加载器.loadClass(),并分析了Class.forName()与ClassLoader的区别。此外,还探讨了通过Class对象进行实例化、获取方法和字段等操作的具体实现。最后从JVM类加载机制角度解析了Class对象的本质及其与类和实例的关系,帮助读者深入理解Java反射的工作原理。
155 0
|
8月前
|
缓存 Dubbo Java
理解的Java中SPI机制
本文深入解析了JDK提供的Java SPI(Service Provider Interface)机制,这是一种基于接口编程、策略模式与配置文件组合实现的动态加载机制,核心在于解耦。文章通过具体示例介绍了SPI的使用方法,包括定义接口、创建配置文件及加载实现类的过程,并分析了其原理与优缺点。SPI适用于框架扩展或替换场景,如JDBC驱动加载、SLF4J日志实现等,但存在加载效率低和线程安全问题。
369 7
理解的Java中SPI机制
|
7月前
|
存储 Java 编译器
Java 中 .length 的使用方法:深入理解 Java 数据结构中的长度获取机制
本文深入解析了 Java 中 `.length` 的使用方法及其在不同数据结构中的应用。对于数组,通过 `.length` 属性获取元素数量;字符串则使用 `.length()` 方法计算字符数;集合类如 `ArrayList` 采用 `.size()` 方法统计元素个数。此外,基本数据类型和包装类不支持长度属性。掌握这些区别,有助于开发者避免常见错误,提升代码质量。
698 1