java反射机制

简介: 了解java反射机制

一、java反射机制

首先大家应该先了解两个概念,编译期和运行期,编译期就是编译器帮你把源代码翻译成机器能识别的代码,比如编译器把java代码编译成jvm识别的字节码文件,而运行期指的是将可执行文件交给操作系统去执行,JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制;
简单说,反射机制指得是程序在运行时能够获取自身的信息。在java中,只要给定类的名字(类的全路径),那么就可以通过反射机制来获得类的所有信息。

大家都知道,要让Java程序能够运行,那么就得让Java类要被Java虚拟机加载。Java类如果不被Java虚拟机加载,是不能正常运行的。不使用反射的话,我们运行的所有的程序都是在编译期的时候就已经知道了你所需要的那个类已经被加载了。

Java的反射机制是在编译并不确定是哪个类被加载了,而是在程序运行的时候才加载、探知、自审。使用在编译期并不知道的类,这样的特点就是反射。

二、反射作用

  1. 假如我们有两个程序员,一个程序员在写程序的时候,需要使用第二个程序员所写的类,但第二个程序员并没完成他所写的类。那么第一个程序员的代码能否通过编译呢?这是不能通过编译的。利用Java反射的机制,就可以让第一个程序员在没有得到第二个程序员所写的类的时候,来完成自身代码的编译。

  2. 反射可以节约JVM资源(但是相对于new对象来说,性能有损失);如果不使用反射,使用普通的new对象的方式,则当前类在加载的时候就会为这个new出来的对象生成.class,占用JVM资源;但是如果使用反射的话,类在加载时不会创建这个对象的.class,不会占用JVM资源,只有当程序运行时,运行到反射这里的时候,才会为这个类生成.class文件;

为什么使用反射,直接使用new对象不就可以了吗?

  • 静态编译:在编译时确定类型,绑定对象。
  • 动态编译:运行时确定类型,绑定对象。动态编译最大限度发挥了java的灵活性,体现了多
    态的应用,用以降低类之间的藕合性。

    一句话,反射机制的优点就是可以实现动态创建对象和编译,体现出很大的灵活性,特别是在J2EE的开发中,它的灵活性就表现的十分明显。比如,一个大型的软件,不可能一次就把把它设计的很完美,当这个程序编译后,发布了,当发现需要更新某些功能时,我们不可能要用户把以前的卸载,再重新安装新的版本,假如这样的话,这个软件肯定是没有多少人用的。采用静态的话,需要把整个程序重新编译一次才可以实现功能的更新,而采用反射机制的话,它就可以不用卸载,只需要在运行时才动态的创建和编译,就可以实现该功能。

    它的缺点是对性能有影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它满足我们的要求。这类操作总是慢于只直接执行相同的操作。 
    

new对象和反射创建的对象有什么区别

  • new;静态编译,在编译期就将模块编译进来,执行该字节码文件,所有的模块都被加载;

  • 反射:动态编译,编译期没有加载,等到模块被调用时才加载;

注:spring的ioc就用到反射机制,newInstance创建。更加的通用,并且降低耦合,避免了硬编码,只需提供一个字符串就可以动态的创建。

什么时候使用反射:

  • Java编码时知道类和对象的具体信息,此时直接对类和对象进行操作即可,无需反射
  • 如果编码时不知道类或者对象的具体信息,此时应该使用反射来实现,去动态的创建对象

反射优缺点

  • 优点
  1. 反射提高了Java程序的灵活性和扩展性,降低耦合性,提高自适应能力。它允许程序创建和控制任何类的对象,无需提高硬编码目标类
  2. 反射是其他一些常用语言,如C、C++、Fortran或者Pascal等不具备的
  3. Java反射技术应用领域很广,如软件测试、JavaBean等
  4. 许多流行的开源框架例如Struts、Hibernate、SpringIOC(配置文件里的类的全路径...)在实现过程中都采用了该技术
  • 缺点
  1. 性能问题:使用反射基本上是一种解释操作(通知jvm要做的事情,性能比直接的java代码要慢很多),用于字段和方法接入时要远慢于直接代码。因此Java反射机制主要应用在对灵活性和扩展性要求很高的系统框架上,普通程序不建议使用;反射包括了一些动态类型,所以JVM无法对这些代码进行优化。因此,反射操作的效率要比那些非反射操作低得多。我们应该避免在经常被执行的代码或对性能要求很高的程序中使用反射。
  2. 使用反射会模糊程序内部逻辑:程序员希望在代码中看到程序的逻辑,反射等绕过了源代码的技术,因而会带来维护问题。反射代码比相应的直接代码更复杂
  3. 安全问题:由于反射允许代码执行一些在正常情况下不被允许的操作(比如访问私有的属性和方法),所以使用反射可能会导致意料之外的副作用;

三、反射的具体实现

3.1、反射的常用操作

通过反射获取类的构造器来创建对象

    //参数是字符串,必须是Person类的全路径
    Class c3 = Class.forName("cn.maltose.demo1.Person");

通过反射获取构造方法,创建对象(前提是有public修饰的空参构造方法)

    //1、获取到Class对象
    Class c = Class.forName("cn.maltose.demo1.Person");//包名.类名
    //2、获取指定的构造方法
    //下边的代码可以获取到这个构造参数:public Person()
    //Constructor con = c.getConstructor(null);

    //下边的代码可以获取到这个构造参数:public Person(String name, int age, String address)
    Constructor con = c.getConstructor(String.class, int.class, String.class);

    //3、通过构造方法类中Constructor的方法,创建对象
    //Object obj = con.newInstance(null);
    Object obj = con.newInstance("小明", 22, "哈尔滨");

    //显示
    System.out.println(obj);

通过反射获取私有构造方法,创建对象

    Class c=Class.forName("cn.maltose.demo1.Person");
    //Person类里的该构造方法是私有的:private Person(String name, int age)
    Constructor con = c.getDeclaredConstructor(String.class, int.class);
    //暴力反射 ,取消 Java 语言访问检查
    con.setAccessible(true);
    //创建对象
    Object obj = con.newInstance("小明",12);
    System.out.println(obj);
    //输出:cn.maltose.demo1.Person@17494c8

通过反射获取成员变量并使用
在反射机制中,把类中的成员变量使用类Field表示。可通过Class类中提供的方法获取成员变量:

    // 获取Class对象
    Class c = Class.forName("cn.maltose.demo1.Person");

    // 获取成员变量, 多个变量
    // Field[] fields = c.getFields();
    //获取所有属性,包括私有属性
    Field[] fields = c.getDeclaredFields();
    for (Field field : fields) {
   
        System.out.println(field);
    }
    System.out.println("-----------------");
    // 一个变量
    //获取公有属性: public int age;
    Field ageField = c.getField("age");
    System.out.println(ageField);

    //获取私有属性: private String address
    Field addressField = c.getDeclaredField("address");
    //暴力反射
    addressField.setAccessible(true);
    System.out.println(addressField);

Person类:

public class Person {
   
    String name = "zhangsan";
    public int age = 20;
    private String address;
}

结果:

    java.lang.String cn.maltose.demo1.Person.name
    public int cn.maltose.demo1.Person.age
    private java.lang.String cn.maltose.demo1.Person.address
    -----------------------------------------
    public int cn.maltose.demo1.Person.age
    private java.lang.String cn.maltose.demo1.Person.address

通过反射,创建对象,获取指定的成员变量,进行赋值与获取值操作
Person类:

public class Person {
   
    public String name;
    public int age;
    private String address;
    public Person() {
   
        super();
    }
    public Person(String name) {
   
        super();
    }
    public Person(String name, int age, String address) {
   
        super();
        this.name = name;
        this.age = age;
        this.address = address;
    }
}

Test类:

    //1,获取Class对象
    Class c = Class.forName("cn.maltose.demo1.Person");
    //2,获取构造方法
    //public Person(String name) 
    Constructor con = c.getConstructor(String.class);
    //3,通过构造方法,创建对象
    Object obj = con.newInstance("小明");
    //4,获取指定的成员变量
    //public String name;
    Field nameField = c.getField("name");
    //public int age;
    Field ageField = c.getField("age");
    //private String address;
    Field addressField = c.getDeclaredField("address");
    addressField.setAccessible(true); //取消 Java 语言访问检查

    //5,通过方法,给指定对象的指定成员变量赋值或者获取值
    System.out.println("赋值前:");
    System.out.println("name = "+ nameField.get(obj));
    System.out.println("age = "+ ageField.get(obj));
    System.out.println("address = "+ addressField.get(obj));

    //赋值
    nameField.set(obj, "张三");
    ageField.set(obj, 23);
    addressField.set(obj, "凯利广场");

    System.out.println("------------------------");
    System.out.println("赋值后:");
    System.out.println("name = "+ nameField.get(obj));
    System.out.println("age = "+ ageField.get(obj));
    System.out.println("address = "+ addressField.get(obj));

结果:

    赋值前:
    name = null
    age = 0
    address = null
    ------------------------
    赋值后:
    name = 张三
    age = 23
    address = 凯利广场

通过反射获取成员方法并使用
在反射机制中,把类中的成员方法使用类Method表示。可通过Class类中提供的方法获取成员方法
Person类:

public class Person {
   
    public String name;
    public int age;
    private String address;
    public Person() {
   
        super();
    }
    public Person(String name) {
   
        super();
    }
    public Person(String name, int age, String address) {
   
        super();
        this.name = name;
        this.age = age;
        this.address = address;
    }
    public String getName() {
   
        return name;
    }
    public void setName(String name) {
   
        this.name = name;
    }
    public int getAge() {
   
        return age;
    }
    public void setAge(int age) {
   
        this.age = age;
    }
    public String getAddress() {
   
        return address;
    }
    public void setAddress(String address) {
   
        this.address = address;
    }
    public void method1(){
   
        System.out.println("Person里的公有method1方法");
    }
    public String method4(String name){
   
        System.out.println("Person里的公有method4(String name)方法");
        return name;
    }
    private void method5(){
   
        System.out.println("Person里的私有method5方法");
    }
}

测试类:

    // 获取Class对象
    Class c = Class.forName("cn.maltose.demo1.Person");
    // 获取多个公共方法,会显示父类方法
    // Method[] methods = c.getMethods();
    //获取所有方法,包括私有方法,不显示父类方法
    Method[] methods = c.getDeclaredMethods();
    System.out.println("获取所有方法:");
    for (Method method : methods) {
   
        System.out.println(method);
    }
    System.out.println("-----------------------");
    // 获取一个方法:
    // public void method1()
    System.out.println("获取一个指定方法:");
    Method method = c.getMethod("method1", null);
    System.out.println(method);
    // public String method4(String name){
   
    method = c.getMethod("method4", String.class);
    System.out.println(method);
    // 私有方法
    // private void method5()
    method = c.getDeclaredMethod("method5", null);
    System.out.println(method);

结果:

    获取所有方法:
    public void cn.maltose.demo1.Person.method1()
    public void cn.maltose.demo1.Person.setAge(int)
    public java.lang.String cn.maltose.demo1.Person.method4(java.lang.String)
    public int cn.maltose.demo1.Person.getAge()
    private void cn.maltose.demo1.Person.method5()
    public java.lang.String cn.maltose.demo1.Person.getName()
    public void cn.maltose.demo1.Person.setName(java.lang.String)
    public java.lang.String cn.maltose.demo1.Person.getAddress()
    public void cn.maltose.demo1.Person.setAddress(java.lang.String)
    -----------------------
    获取一个指定方法:
    public void cn.maltose.demo1.Person.method1()
    public java.lang.String cn.maltose.demo1.Person.method4(java.lang.String)
    private void cn.maltose.demo1.Person.method5()

通过反射,创建对象,调用指定的方法

      //1, 获取Class对象
    Class c = Class.forName("cn.maltose.demo1.Person");
    //2,获取构造方法
    //public Person(String name, int age, String address){
   
    Constructor con = c.getConstructor(String.class, int.class, String.class);
    //3,通过构造方法,创建对象
    Object obj = con.newInstance("小明", 23, "哈尔滨");
    //4,获取指定的方法
    //public void method1()  没有返回值没有参数的方法
    //Method m1 = c.getMethod("method1", null);//null可以不写

    //public String method4(String name)
    Method m4 = c.getMethod("method4", String.class);

    //5,执行找到的方法
    //m1.invoke(obj, null);

    Object result = m4.invoke(obj, "itcast");
    System.out.println("result = " + result);

通过反射,创建对象,调用指定的private 方法:

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