我正在使用Oracle数据库,我在插入之前有序列和触发器来生成和存储ID。
CREATE SEQUENCE CASE_SEQ START WITH 1001 INCREMENT BY 1 NOMAXVALUE;
CREATE OR REPLACE TRIGGER CASE_TR_SEQ
BEFORE INSERT ON CASE FOR EACH ROW
BEGIN
SELECT CASE_SEQ.NEXTVAL INTO :NEW.CASE_ID FROM DUAL;
END;
/
然后我有一个带有属性的简单实体:
@Id
@Column(name = "CASE_ID", insertable = false, updatable = false)
private Long caseId;
...当我尝试构建项目时,我得到了:
Exception [EclipseLink-46] (Eclipse Persistence Services - 2.3.2.v20111125-r10461):
org.eclipse.persistence.exceptions.DescriptorException
Exception Description: There should be one non-read-only mapping defined for the
primary key field [CASE.CASE_ID].
当我删除可插入或可更新的关键字时,它可以工作。我知道有很多解决方案如何使用JPA生成ID,JPA也可以使用(调用)oracle序列来设置(生成)ID。但我试着理解为什么我的解决方案是错误的。为什么我不能将这两个关键字与@Id注释一起使用?我的想法是:我想禁止JPA插入或更新caseId。
1)什么是正确的洗液?我应该只使用@Id:
@Id
@Column(name = "CASE_ID")
private Long caseId;
或更好(更安全)定义insertable = false:
@Id
@Column(name = "CASE_ID", insertable = false)
private Long caseId;
2)我理解@Id的updatable = false没有意义(更新主键没有意义但是原始sql可能有意义),但是它意味着什么(你有什么好处的例子吗? ):
@Id
@Column(name = "CASE_ID", updatable = false)
private Long caseId;
我做了一些测试:
实体
@Id
@Column(name = "CASE_ID")
private Long caseId;
JPA日志
INSERT INTO CASE (CASE_ID, CASE_DATE, INFO) VALUES (?, ?, ?)
bind => [3 parameters bound]|#]
因此这不安全,因为JPA尝试存储CASE_ID(然后由触发器中的Oracle序列替换为ID)。
实体
@Id
@Column(name = "CASE_ID", insertable = false)
private Long caseId;
创建方法
public void createCase(final Case caseData) {
caseData.setCaseId(-1001L);
em.persist(caseData);
}
JPA日志
INSERT INTO CASE (CASE_DATE, INFO) VALUES (?, ?)
bind => [2 parameters bound]|#]
这很好,因为CASE_ID不是插入命令的一部分。
由于注释ID:
,因此无法更新CASE_IDpublic void update() {
Case update = em.find(Case.class, 1023L);
update.setCaseId(1028L);
}
Exception [EclipseLink-7251] (Eclipse Persistence Services - 2.3.2.v20111125-r10461):
org.eclipse.persistence.exceptions.ValidationException
Exception Description: The attribute [caseId] of class
[com.wordpress.kamiluv.jsfprototype.model.entity.Case] is mapped to a primary
key column in the database. Updates are not allowed.
所以现在最后一个版本看起来最安全,对吧?
答案 0 :(得分:5)
您目前正在使用JPA注释说明您有一个无法以任何方式插入或更新的@Id列。您必须能够在插入或更新之前设置ID,但JPA不知道如何执行此操作。如果你不想在代码中设置id,你需要在@Id上使用@GeneratedValue注释告诉JPA使用什么策略(TABLE,SEQUENCE,IDENTITY,AUTO之一)。