java中newInstance()和new()

本文详细解释了Java中Class.forName()方法的用途及其与newInstance()的区别,并探讨了这两种方法在数据库开发中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


在Java开发特别是数据库开发中,经常会用到Class.forName( )这个方法。通过查询Java Documentation我们会发现使用Class.forName( )静态方法的目的是为了动态加载类。在加载完成后,一般还要调用Class下的newInstance( )静态方法来实例化对象以便操作。因此,单单使用Class.forName( )是动态加载类是没有用的,其最终目的是为了实例化对象。

这里有必要提一下就是Class下的newInstance()和new有什么区别?,首先,newInstance( )是一个方法,而new是一个关键字,其次,Class下的newInstance()的使用有局限,因为它生成对象只能调用无参的构造函数,而使用new关键字生成对象没有这个限制。
好,到此为止,我们总结如下:
Class.forName("")返回的是类
Class.forName("").newInstance()返回的是object
有数据库开发经验朋友会发现,为什么在我们加载数据库驱动包的时候有的却没有调用newInstance( )方法呢?即有的jdbc连接数据库的写法里是Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance(),为什么会有这两种写法呢?
刚才提到,Class.forName("");的作用是要求JVM查找并加载指定的类,如果在类中有静态初始化器的话,JVM必然会执行该类的静态代码段。而在JDBC规范中明确要求这个Driver类必须向DriverManager注册自己,即任何一个JDBC Driver的Driver类的代码都必须类似如下:

public class MyJDBCDriver implements Driver {
	static {
		DriverManager.registerDriver(new MyJDBCDriver());
	}
}
既然在静态初始化器的中已经进行了注册,所以我们在使用JDBC时只需要Class.forName(XXX.XXX);就可以了。
Java中工厂模式经常使用newInstance()方法来创建对象,因此从为什么要使用工厂模式上可以找到具体答案。 例如:
class c = Class.forName("Example");
factory = (ExampleInterface)c.newInstance();
其中ExampleInterface是Example的接口,可以写成如下形式:
String className = "Example";
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
进一步可以写成如下形式:
String className = readfromXMlConfig;//从xml 配置文件中获得字符串
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
上面代码已经不存在Example的类名称,它的优点是,无论Example类怎么变化,上述代码不变,甚至可以更换Example的兄弟类Example2 , Example3 , Example4……,只要他们继承ExampleInterface就可以。
从JVM的角度看,我们使用关键字new创建一个类的时候,这个类可以没有被加载。但是使用newInstance()方法的时候,就必须保证:1、这个 类已经加载;2、这个类已经连接了。而完成上面两个步骤的正是Class的静态方法forName()所完成的,这个静态方法调用了启动类加载器,即加载 java API的那个加载器。
现在可以看出,newInstance()实际上是把new这个方式分解为两步,即首先调用Class加载方法加载某个类,然后实例化。 这样分步的好处是显而易见的。我们可以在调用class的静态加载方法forName时获得更好的灵活性,提供给了一种降耦的手段。
最后用最简单的描述来区分new关键字和newInstance()方法的区别:
newInstance: 弱类型。低效率。只能调用无参构造。
new: 强类型。相对高效。能调用任何public构造。

<think>我们正在讨论JavaClass类的newInstance方法。用户之前提到了两种创建对象的方式:newClass.newInstance,现在想深入理解newInstance方法的作用用法。 注意:用户提供的引用中有一个示例代码:CreateInstance instance = (CreateInstance) Class.forName("com.create.instance.CreateInstance").newInstance(); 我们将围绕这个示例展开讲解。 首先,我们需要明确:Class.newInstance()Java反射机制中用于创建对象的一种方式。它与new关键字创建对象的主要区别在于: 1. new是静态编译时确定,而newInstance()是动态运行时创建。 2. newInstance()要求类必须有一个无参构造器,且该构造器的访问权限要足够。 接下来,我们将详细解释newInstance()方法。 步骤: 1. 方法定义:Class类的newInstance()方法用于创建该Class对象所表示类的一个新实例。它相当于调用该类的无参构造函数。 2. 使用条件: - 类必须有一个无参数的构造方法。 - 类的构造方法的访问权限要足够,如果构造方法是私有的,则需要在调用newInstance()之前设置可访问性(通过setAccessible(true))。 3. 使用步骤: a. 获取Class对象:可以通过Class.forName("全限定类名")来获取。 b. 调用newInstance()方法创建实例。 c. 由于newInstance()返回的是Object类型,需要强制类型转换。 示例代码(基于用户引用中的代码): CreateInstance instance = (CreateInstance) Class.forName("com.create.instance.CreateInstance").newInstance(); 4. 异常处理:newInstance()方法可能抛出以下异常: - InstantiationException:如果该类是一个抽象类、接口、数组类、基本类型或者没有无参构造器,则抛出此异常。 - IllegalAccessException:如果类或者其无参构造方法不可访问(如private且没有设置可访问),则抛出此异常。 - ClassNotFoundException(由forName方法抛出):如果找不到指定的类。 因此,通常需要捕获这些异常。 5.new关键字的对比: - new:在编译时确定,直接调用构造方法。 - newInstance():在运行时动态加载类并创建对象,常用于框架设计(如Spring的IoC容器)动态代理。 6. 注意:从Java 9开始,Class.newInstance()方法被标记为过时(deprecated),推荐使用Constructor类的newInstance()方法。因为Constructor.newInstance()可以调用有参数的构造方法,并且可以处理受检异常。 替代方法(Java 9+): // 获取Class对象 Class<?> clazz = Class.forName("com.create.instance.CreateInstance"); // 获取指定的构造器(无参) Constructor<?> constructor = clazz.getDeclaredConstructor(); // 如果是私有构造器,需要设置可访问 constructor.setAccessible(true); // 创建实例 CreateInstance instance = (CreateInstance) constructor.newInstance(); 7. 使用场景: - 动态加载类(例如,根据配置文件中的类名创建对象) - 工厂模式中创建对象 - 框架设计(如Spring通过反射创建Bean) 8. 示例代码(包含异常处理): try { // 加载类 Class<?> clazz = Class.forName("com.create.instance.CreateInstance"); // 使用Class.newInstance(注意:Java 9+已过时) CreateInstance instance1 = (CreateInstance) clazz.newInstance(); // 使用Constructor.newInstance(推荐) Constructor<?> constructor = clazz.getDeclaredConstructor(); constructor.setAccessible(true); // 如果构造器是私有的,需要设置 CreateInstance instance2 = (CreateInstance) constructor.newInstance(); } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | NoSuchMethodException | InvocationTargetException e) { e.printStackTrace(); } 9. 注意事项: - 使用newInstance()时,如果类没有无参构造器,会抛出InstantiationException。因此,确保类有无参构造器(或者使用Constructor的newInstance并传入参数)。 - 对于私有构造器,需要设置setAccessible(true)来覆盖访问控制,否则会抛出IllegalAccessException。 总结: newInstance()是反射机制中创建对象的基本方法,但已被Constructor.newInstance()取代。它的核心价值在于运行时动态创建对象,为框架动态加载提供支持。 相关问题: 1. Java反射机制中如何调用有参数的构造方法? 2. 为什么Java 9废弃了Class.newInstance()方法? 3. 如何使用反射创建数组对象? 4. 反射创建对象new创建对象在性能上有什么差异?</think>### Java中`Class.newInstance()`方法详解 #### 一、核心作用 `Class.newInstance()`是Java反射机制的核心方法,用于**动态创建类的实例对象**。与`new`关键字相比,它实现了: 1. **运行时动态加载**:根据类名字符串创建对象 2. **解耦设计**:调用方无需在编译时知道具体类 3. **框架支持**:Spring等框架的IoC容器底层依赖此机制[^1] #### 二、使用条件 必须同时满足: ```java class MyClass { // 1. 必须存在无参构造器 public MyClass() {} // 2. 构造器访问权限足够 // (非private且非受限包权限) } ``` #### 三、标准使用步骤 ```java try { // 1. 通过全限定类名加载Class对象 Class<?> clazz = Class.forName("com.example.MyClass"); // 2. 调用newInstance()创建实例 Object instance = clazz.newInstance(); // 3. 强制类型转换 MyClass myObj = (MyClass) instance; } catch (ClassNotFoundException e) { // 类不存在异常 } catch (InstantiationException e) { // 构造器不可访问/抽象类异常 } catch (IllegalAccessException e) { // 无权限访问构造器 } ``` #### 四、与`new`关键字的对比 | 特性 | `new`关键字 | `Class.newInstance()` | |---------------------|----------------------|-----------------------------| | 编译时依赖 | 需要类可见 | 只需类名字符串 | | 构造器要求 | 可调用任意构造器 | 必须有无参构造器 | | 性能 | 高(直接JVM支持) | 较低(反射开销) | | 典型场景 | 常规对象创建 | 框架/动态加载 | | 异常处理 | 无检查异常 | 需处理3类检查异常 | #### 五、Java 9+的替代方案 由于`Class.newInstance()`在Java 9被标记为过时,推荐使用: ```java // 使用Constructor.newInstance() Constructor<MyClass> constructor = MyClass.class.getConstructor(); MyClass obj = constructor.newInstance(); // 更灵活,可处理带参构造 ``` #### 六、典型应用场景 1. **动态加载插件** ```java String pluginClass = config.getProperty("plugin"); Plugin p = (Plugin) Class.forName(pluginClass).newInstance(); p.execute(); ``` 2. **工厂模式实现** ```java public class ObjectFactory { public static Object create(String className) { return Class.forName(className).newInstance(); } } ``` 3. **数据库驱动加载** (JDBC经典用法) ```java Class.forName("com.mysql.cj.jdbc.Driver").newInstance(); ``` #### 七、注意事项 1. **初始化问题**:不会执行静态代码块(需显式调用`Class.forName()`) 2. **性能优化**:高频调用场景应缓存`Class`对象 3. **安全限制**:受`SecurityManager`管控,可能抛出`SecurityException` > **反射性能基准测试**:在JDK 17中,`newInstance()`比`new`慢约5倍,但通过缓存`Constructor`对象可缩减至2倍差距[^1]。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值