new 是 Java 最常用实例化方式,触发构造方法、分配堆内存并返回引用;必须带括号调用构造方法,无参构造仅在未定义任何构造时默认提供;反射、clone、反序列化和工厂方法为替代方案,各具语义与限制。

new 关键字是最常用也是最直接的实例化方式
Java 中创建对象的核心语法就是 new,它会触发类的构造方法、分配堆内存、返回引用。几乎所有标准场景都从这里开始。
常见错误现象:忘记写 ()(如 new String 而非 new String())会导致编译失败;调用无参构造时括号可省略?不,String s = new String; 是非法语法,必须带括号。
-
new后必须跟类名 + 括号(即使无参),括号内可传参匹配对应构造方法 - 若类没有显式定义任何构造方法,编译器自动提供无参默认构造;一旦定义了带参构造,默认构造就消失,此时
new MyClass()会编译报错 - 构造方法里抛出异常(如
IOException)时,new表达式所在行必须处理该异常(try-catch 或 throws)
Person p = new Person("Alice", 30);
List list = new ArrayList<>(); // Java 7+ 支持菱形运算符
Class.newInstance() 已被弃用,别再用
在 Java 9 中,Class.newInstance() 被标记为 @Deprecated(forRemoval = true),运行时可能抛 InstantiationException 或绕过访问控制,安全隐患大。
替代方案是使用 Constructor.newInstance(),它支持私有构造、参数类型检查、异常包装更清晰。
立即学习“Java免费学习笔记(深入)”;
-
Class.forName("com.example.User").newInstance()—— 错误写法,已淘汰 - 正确做法:
Constructorctor = User.class.getDeclaredConstructor(String.class); ctor.setAccessible(true); User u = ctor.newInstance("test"); - 注意:反射创建对象性能较低,且需处理
InvocationTargetException、NoSuchMethodException等一堆受检异常
clone() 方法不是构造,但能产生新对象
clone() 不调用任何构造方法,而是对当前对象做浅拷贝(字段值逐位复制)。要让它工作,类必须实现 Cloneable 接口并重写 clone(),否则抛 CloneNotSupportedException。
典型陷阱:数组、集合、自定义引用类型字段不会被深拷贝,修改副本会影响原对象。
- 浅拷贝示例:
User u2 = u1.clone();→u2.name和u1.name指向同一字符串(不可变所以安全),但u2.address若是Address对象引用,则和u1.address共享同一实例 - 需要深拷贝时,必须在重写的
clone()方法中手动 new 新对象并复制内部状态 -
clone()的语义模糊、API 设计反直觉,现代代码中更倾向用构造函数或 builder 模式替代
@Override
protected Object clone() throws CloneNotSupportedException {
User cloned = (User) super.clone();
cloned.address = new Address(this.address.city); // 手动深拷贝
return cloned;
}
反序列化和工厂方法属于间接实例化,行为差异大
通过 ObjectInputStream.readObject() 创建对象时,**完全不执行任何构造方法**,JVM 直接分配内存并填充字段值(包括 private 和 final 字段),这是 JVM 层面的特殊机制。
工厂方法(如 Calendar.getInstance()、LocalDateTime.now())则封装了 new 或其他逻辑,对外隐藏实现细节,利于替换策略或复用对象(如单例池)。
- 反序列化要求类实现
Serializable,且最好定义serialVersionUID;若含不可序列化字段,需加transient或自定义readObject() - 工厂方法返回的可能是子类实例(如
getInstance()返回BuddhistCalendar),也可能是缓存对象(如Boolean.valueOf(true)返回常量池中已有引用) - 不要假设工厂方法一定返回新对象——
Integer.valueOf(100)和Integer.valueOf(200)行为不同(后者新建,前者可能复用)
new 就意味着你掌控初始化逻辑;用反射就得承担安全与异常成本;用 clone() 得自己厘清深浅拷贝边界;而反序列化和工厂方法背后,往往藏着框架约定或性能优化意图。写代码时看到 new,不妨多问一句:这里真需要一个全新对象吗?










