我正在努力将 jpa 实体转换为使用 lombok。结果代码如下:
@Entity
@Table(name = "TEST")
@Data
@NoArgsConstructor
@AllArgsConstructor
class Test {
...
@Column(name = "FORMATTING")
@Enumerated(EnumType.ORDINAL)
private FormatType formatType;
...
}
生成的错误消息包含以下内容
Caused by: org.hibernate.HibernateException: Missing column: formatType in TEST
我真的不知道要在这里谷歌搜索什么。 (我之前尝试粘贴所有内容formatType
进入谷歌 - 没有看到任何东西)
NOTE:
为了简洁和隐私起见,字段已被重命名,并且似乎不相关的方面已被省略。如果某些内容看起来像是拼写错误,那么很可能就是这样。如果您发现任何问题,请告诉我,以便我解决。
描述该字段的 3 行与我正在使用的代码相比没有变化
EDIT:
我刚刚在错误消息出现之前注意到这一点
13:22:19,967 INFO [org.hibernate.tool.hbm2ddl.TableMetadata] (ServerService Thread Pool -- 57) HHH000261: Table found: TABLE
13:22:19,967 INFO [org.hibernate.tool.hbm2ddl.TableMetadata] (ServerService Thread Pool -- 57) HHH000037: Columns: [..., formatType, ...]
13:22:19,968 ERROR [org.jboss.msc.service.fail] (ServerService Thread Pool -- 57) MSC000001: Failed to start service jboss.persistenceunit."...": org.jboss.msc.service.StartException in service jboss.persistenceunit."...": javax.persistence.PersistenceException: [PersistenceUnit: ...] Unable to build EntityManagerFactory
应该是功能性的
@Entity
@Inheritance(strategy = InheritanceType.JOINED)
@Table(name = "PARENT")
public abstract class Parent implements Serializable {
private static final long serialVersionUID = 1;
@Id
@Column(name = "ID")
@GeneratedValue
private long id;
@Column(name = "ENABLED")
private boolean enabled;
}
@Entity
@Table(name = "CHILD")
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Child extends Parent {
/** XXX: HERE BE DRAGONS */
@Column(name = "ENUM_1")
@Enumerated(EnumType.STRING)
private Enum1 enum1;
@Column(name = "ENUM_2")
@Enumerated(EnumType.ORDINAL)
private Enum2 enum2;
/** XXX: NO MORE DRAGONS */
@Column(name = "FREQUENCY")
private String frequency;
@Column(name = "EMPTY")
private boolean empty;
@Column(name = "MAX_SIZE")
private int maxSize;
}
public enum Enum1 {
A,
B,
C
}
public enum Enum2 {
X,
Y,
Z
}
我已经回滚了 lombok 更改,我仍然想知道问题是什么,但并不着急。另外,由于这个可爱的小错误,我大约落后了 4 个小时,所以我的回复可能有点慢。
子表的 pk 是父表的 fk,如果没有 lombok,一切似乎都可以工作,尽管事实上Child
班级没有 id。
解决方案:
我完全忘记问这个了。不久前我又重新审视了这个问题。为了解释该解决方案,让我们看一下我包含的第一个示例的稍微简化的版本。
@Entity
@Table(name = "TEST")
@Setter
@Getter
class Test {
...
@Column(name = "FORMATTING")
@Enumerated(EnumType.ORDINAL)
private FormatType formatType;
...
}
Lombok 似乎会给你这个:
@Entity
@Table(name = "TEST")
class Test {
...
@Column(name = "FORMATTING")
@Enumerated(EnumType.ORDINAL)
private FormatType formatType;
public FormatType getFormatType() {
return formatType;
}
public void setFormatType(FormatType formatType) {
this.formatType = formatType;
}
...
}
请注意,注释仍然附加在field。现在,我不确定这是否只是我们正在使用的 JPA 的版本或实现,但我认为如果定义了访问器,jpa 只会忽略除此之外的任何注释@Column
(以及为@Column
- 这就是为什么 jpa 寻找错误的列名称)。所以我们实际上需要:
@Entity
@Table(name = "TEST")
class Test {
...
private FormatType formatType;
@Column(name = "FORMATTING")
@Enumerated(EnumType.ORDINAL)
public FormatType getFormatType() {
return formatType;
}
public void setFormatType(FormatType formatType) {
this.formatType = formatType;
}
...
}
经过大量的困惑,试图找到示例并填写一些有关 lombok 如何执行其操作的细节(公平地说,我很容易混淆),我发现了这个小宝石:onMethod=@__({@AnnotationsHere}) https://projectlombok.org/features/experimental/onX。利用这个功能,我想出了以下内容:
@Entity
@Table(name = "TEST")
@Setter
class Test {
...
@Getter(onMethod=@__({
@Column(name = "FORMATTING"),
@Enumerated(EnumType.ORDINAL)
}))
private FormatType formatType;
...
}
很快就起作用了。现在我们有了显然是唯一可用的解决方案,我想解决我们目前都在思考的问题:这真的比手动编写方法并在其中附加注释更干净吗?回答:……我不知道。我很高兴我找到了解决方案。