18

为什么我们需要在许多 Java 相关 API 中使用默认的无参数构造函数?就像一般规则一样,所有 java bean 类或实体类(JPA 等)或 JAX-WS 实现类都需要显式的无参数构造函数。

如果默认情况下 Java 提供了无参数构造函数,那么为什么这些标准中的大多数都需要显式构造函数?

4

6 回答 6

21

如果没有定义其他构造函数, Java 仅提供默认的无参数构造函数。因此,如果您有其他构造函数,您必须自己显式定义一个无参数构造函数。

这些框架使用反射 API 并查看方法名称来确定如何设置属性。构造函数的参数只能按类型而不是名称找到,因此框架无法可靠地将属性与构造函数参数匹配。因此,他们需要一个无参数的构造函数来创建对象,然后可以使用 setter 方法来初始化数据。

一些框架可能支持@ConstructorProperties作为替代方案。

于 2010-12-20T14:14:01.877 回答
11

我相信需要publicnullary 构造函数的框架这样做是因为它们使用反射来实例化类型,例如通过Class.newInstance().

至于为什么默认构造函数可能不适用于这种情况,这里是相关的 JLS 部分:

JLS 8.8.9 默认构造函数

如果 aclass不包含构造函数声明,则自动提供不带参数的默认构造函数:

  • 如果class声明了public,则默认构造函数被隐式地赋予访问修饰符public
  • 如果类被声明protected,则默认构造函数被隐式地赋予访问修饰符protected
  • 如果类被声明private,则默认构造函数被隐式地赋予访问修饰符private
  • 否则,默认构造函数具有没有访问修饰符隐含的默认访问权限。

因此,在一个public类中,默认构造函数将具有正确的可见性,但否则public必须明确提供。

于 2010-06-20T06:17:27.550 回答
4

需要构造函数来初始化任何非默认值,并且可以包含所需的副作用。

如果您的编程风格鼓励数据传输对象的最小构造函数,这似乎是不必要的,但是库选择不为构造函数假定任何编程风格。

您可以编写一个不假设默认构造函数的库,但您必须假设构造函数会做什么和不会做什么。即我已经编写了这样的库,但也能够强制构造函数允许做什么,这样不直接调用它是安全的。

于 2010-06-20T06:19:34.473 回答
3

如果没有应用其他构造函数,Java 仅提供无参数构造函数。在许多 Java API(例如 JPA、序列化和许多其他从外部表示构建对象)中,在设置对象的数据值之前需要对象的实例,因为如何应用这些值的定义是通过对象的实例成员定义(例如 readExternal(ObjectInput))。如果一个类只有一个带有一些参数的构造函数,那么除非定义了一个单独的无参数构造函数,否则库可能无法构造一个实例。

值得注意的是,这是特定库的实现者的设计选择,可以构建一个 API/框架,可以外部化和重新创建没有无参数构造函数的对象(定义一个单独的工厂类是一种方法)。需要无参数构造函数的模式首先出现在 Java 序列化(我认为)中,并已被其他库(例如 JPA)采用为事实上的标准方法。这种方法的缺点是它阻止了不可变对象的使用。

于 2010-06-20T06:19:22.837 回答
0

两个原因:1) 如果引用数据类型的实例数据字段未初始化,则避免出现 NullPointerException。提供显式构造函数将使您有机会初始化此类数据字段,如果它们在声明时尚未初始化 2) 对于想要使用无参数构造函数的用户;如果存在其他构造函数,这些不会自动提供

于 2014-01-14T11:17:01.317 回答
0

许多这些框架都源自“ POJO ”的早期思想,尤其是JavaBeans。按照惯例,最小的有用 Java 对象将具有一个无参数构造函数,并且每个成员都可以通过 get/setProperty1 和 is/setProperty1 等方法访问布尔值。如果类遵循接口的约定,则使用反射的工具和框架可以“开箱即用”。

于 2019-03-28T12:44:15.767 回答