Java反射之创建对象

简介: Java反射机制提供了另外一种创建对象方法,Class类提供了一个实例方法newInstance(),通过该方法 可以创建对象,使用起来比较简单,下面两条语句实现了创建字符串String对象。

Java反射机制提供了另外一种创建对象方法,Class类提供了一个实例方法newInstance(),通过该方法 可以创建对象,使用起来比较简单,下面两条语句实现了创建字符串String对象。

Class clz = Class.forName("java.lang.String");
 String str = (String) clz.newInstance();

这两条语句相当于String str = new String()语句。

注意:意newInstance()方法有可以会抛出 InstantiationException和IllegalAccessException异常,InstantiationException不能实例化异常, IllegalAccessException是不能访问构造方法异常。


一、调用构造方法


调用方法newInstance()创建对象,这个过程中需要调用构造方法,上面的代码只是调用了String的默认 构造方法。如果想要调用非默认构造方法,需要使用Constructor对象,它对应着一个构造方法,获得 Constructor对象需要使用Class类的如下方法:


Constructor[] getConstructors():返回所有公有构造方法Constructor对象数组

Constructor[] getDeclaredConstructors():返回所有构造方法Constructor对象数组。

Constructor getConstructor(Class… parameterTypes):根据参数列表返回一个共有

Constructor对 象。参数parameterTypes是Class数组,指定构造方法的参数列表。

Constructor getDeclaredConstructor(Class… parameterTypes):根据参数列表返回一个Constructor对 象。参数parameterTypes同上。

示例代码如下:

public class HelloWorld {
    public static void main(String[] args) {
        try {
            Class clz = Class.forName("java.lang.String");
//            使用String的默认的构造方法public String()
            String str1 = (String) clz.newInstance();
            System.out.println(str1);
//           1.设置构造方法参数类型
            Class[] params = new Class[1];
//            第一个参数是String
            params[0] = String.class;
//           2.获取与参数对应的构造方法
            Constructor constructor = clz.getConstructor(params);
//           3.为构造方法传递参数
            Object[] argObjs = new Object[1];
//            第一个参数传递Hello
            argObjs[0] = "Hello";
//            4.调用非默认构造方法,构造方法第一个参数是String类型
            String str2 = (String) constructor.newInstance(argObjs);
            System.out.println(str2);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

运行结果:

Hello


二、案例:依赖注入实现


Java反射机制能够在运行时动态加载类,而不是在编译期。在一些框架开发中经常将要实例化的类名 保存到配置文件中,在运行时从配置文件中读取类名字符串,然后动态创建对象,建立依赖关系 1 。采 用new创建对象依赖关系是在编译期建立的,反射机制能够将依赖关系推迟到运行时建立,这种依赖 关系动态注入进来称为依赖注入。


例如:如图所示有三个类,Student和Worker继承自Person,在HelloWorld类的main()方法中会创建 Person子类实例,至于依赖哪一个类,Student还是Worker,可以在运行时从配置文件中读取,然后创 建对象。9.png


Person类:

public class Person {
    public String name;
    public int age;
    public Person(String name,int age){
        this.name = name;
        this.age = age;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public int getAge() {
        return age;
    }
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

worker类:

public class Worker extends Person{
    public String factory;
    public Worker(String name, int age,String factory) {
        super(name, age);
        this.factory = factory;
    }
    public void setFactory(String factory) {
        this.factory = factory;
    }
    public String getFactory() {
        return factory;
    }
    @Override
    public String toString() {
        return "Worker{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", factory='" + factory + '\'' +
                '}';
    }
}

Student类:

public class Student extends Person{
    public String school;
    public Student(String name, int age,String school) {
        super(name, age);
        this.school = school;
    }
    public void setSchool(String school) {
        this.school = school;
    }
    public String getSchool() {
        return school;
    }
    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", school='" + school + '\'' +
                '}';
    }
}

写完这三个类之后,在IDEA项目根目录建一个文本文件Configuration.ini,内容如下:

Java反射机制.创建对象.依赖注入实现案例.Student

调用代码如下:

import java.io.*;
import java.lang.reflect.Constructor;
/**
 * @author : 蔡政洁
 * @email :caizhengjie888@icloud.com
 * @date : 2020/2/27
 * @time : 6:42 下午
 */
public class HelloWorld {
    public static void main(String[] args) {
        try{
//            通过调用readClassName()方法从Configuration.ini文件中读取类名
            String classname = readClassName();
//            通过从配置文件Configuration.ini中读取的字符串创建Class对象
            Class clz = Class.forName(classname);
//            1.指定参数类型
            Class[] params = new Class[3];
//            第一个参数是String
            params[0] = String.class;
//            第二个参数是int
            params[1] = int.class;
//            第三个参数是String
            params[2] = String.class;
//            2.获得对应参数的构造方法
            Constructor constructor = clz.getConstructor(params);
//            3.设置传递参数
            Object[] argObjs = new Object[3];
//            第一个参数传递Tony
            argObjs[0] = "Tony";
//            第二个参数传递21
            argObjs[1] = 21;
//            第三个参数传递北京大学
            argObjs[2] = "北京大学";
//            4.采用非默认构造方法
            Object p = constructor.newInstance(argObjs);
            System.out.println(p);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
//    从Configuration.ini文件中读取类名
    public static String readClassName(){
        FileInputStream readfile = null;
        InputStreamReader ir = null;
        BufferedReader in = null;
        try {
            readfile = new FileInputStream("Configuration.ini");
            ir = new InputStreamReader(readfile);
            in = new BufferedReader(ir);
//            读取文件中的一行数据
            String str = in.readLine();
            return str;
        } catch (FileNotFoundException e) {
            System.out.println("处理FileNotFoundException");
            e.printStackTrace();
        } catch (IOException e) {
            System.out.println("处理IOException");
            e.printStackTrace();
        }
        return null;
    }
}

运行结果:

Student{name='Tony', age=21, school='北京大学'}


相关文章
|
12天前
|
Java
java代码优化:判断内聚到实体对象中和构造上下文对象传递参数
通过两个常见的java后端实例场景探讨代码优化,代码不是优化出来的,而是设计出来的,我们永远不可能有专门的时间去做代码优化,优化和设计在平时
29 15
|
25天前
|
Java 数据库连接 Spring
反射-----浅解析(Java)
在java中,我们可以通过反射机制,知道任何一个类的成员变量(成员属性)和成员方法,也可以堆任何一个对象,调用这个对象的任何属性和方法,更进一步我们还可以修改部分信息和。
|
2月前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
2月前
|
监控 Java
Java基础——反射
本文介绍了Java反射机制的基本概念和使用方法,包括`Class`类的使用、动态加载类、获取方法和成员变量信息、方法反射操作、以及通过反射了解集合泛型的本质。同时,文章还探讨了动态代理的概念及其应用,通过实例展示了如何利用动态代理实现面向切面编程(AOP),例如为方法执行添加性能监控。
|
3月前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
73 17
|
2月前
|
Java
Java的反射
Java的反射。
43 2
|
2月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
3月前
|
存储 Java
[Java]反射
本文详细介绍了Java反射机制的基本概念、使用方法及其注意事项。首先解释了反射的定义和类加载过程,接着通过具体示例展示了如何使用反射获取和操作类的构造方法、方法和变量。文章还讨论了反射在类加载、内部类、父类成员访问等方面的特殊行为,并提供了通过反射跳过泛型检查的示例。最后,简要介绍了字面量和符号引用的概念。全文旨在帮助读者深入理解反射机制及其应用场景。
53 0
[Java]反射
|
3月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
40 6
|
3月前
|
Oracle Java 关系型数据库
重新定义 Java 对象相等性
本文探讨了Java中的对象相等性问题,包括自反性、对称性、传递性和一致性等原则,并通过LaptopCharger类的例子展示了引用相等与内容相等的区别。文章还介绍了如何通过重写`equals`方法和使用`Comparator`接口来实现更复杂的相等度量,以满足特定的业务需求。
39 3