我试图将一个对象持久化到数据库中。 一直得到'Column ID不能接受空值错误'。 我的对象是这样的。
@Entity
public class TestTable {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private Integer id = 0;
@Column(nullable=false, length=256)
private String data = "";
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getData() {
return data;
}
public void setData(String data) {
this.data = data;
}
}
我的持久化函数
public static synchronized boolean persistObject(Object obj){
boolean success = true;
EntityManager em = null;
EntityTransaction tx = null;
try{
em = getEmf().createEntityManager();
tx = em.getTransaction();
tx.begin();
em.persist(obj);
tx.commit();
} catch (Exception e){
success = false;
} finally{
try{
em.close();
} catch(Exception e){
//nothing
}
}
return success;
}
你可以使用 GenerationType.TABLE。这样一来,jpa使用序列表进行id分配,你可能永远不需要生成序列或自动增量值或触发器,降低了可移植性。
另外要注意的是,在java中int类型默认是以0发起的,所以你也可以把它去掉。
在我的例子中,这是关于糟糕的方言。
hibernate.dialect=org.hibernate.dialect.H2Dialect
而不是:
hibernate.dialect=org.hibernate.dialect.PostgreSQL9Dialect
当我切换到生产数据库时,Hibernate试图使用策略。Hibernate试图使用为不同db引擎准备的策略。
我也遇到过和你类似表现的问题。我最终发现我的数据库连接的配置是错误的:我连接的是一个旧的数据库,它的模式不正确。新的模式将主键列声明为
"ID" BIGINT NOT NULL GENERATED ALWAYS AS IDENTITY (START WITH 1, INCREMENT BY 1)
所以 数据库本身自动生成主键 而旧的模式则将其声明为
"ID" INTEGER NOT NULL
Hibernate为新的模式执行了正确的代码,而在旧的模式上却失败了,因为旧的模式要求SQL? INSERT
提供一个值 ID
列。
当ID列是Int时,Hibernate会以沉默而神秘的方式失败。试着在代码中把它改成Long,在数据库中改成无符号64位整数。这为我解决了这个问题。
或者尝试用 @GeneratedValue(strategy = GenerationType.AUTO)
而不是 @GeneratedValue(strategy = GenerationType.SEQUENCE)
.
以防有人到了这个帖子里和我遇到同样的问题。我一直在努力解决这个问题,现在已经有几个小时了,我只想把我的解决方案贴出来。似乎ID总是被设置为0,这引发了一个错误。
模型中的id属性必须是 "long "或 "Integer "类型,而不是 "long "或 "integer"(注意大写)。我放的是 "long",这是一个基元类型,不能为空。因此默认值是0,Hibernate没有意识到这是一个新的实体,于是试图用ID=0保存它,这显然是行不通的。
将其改成可以为空的引用类型 "Long",就解决了这个问题。
我也在为同样的问题而苦恼,不知何故,对我来说,long类型比long类型更有效:)。作为一个变通办法,我把数据库中的id列转换为SERIEL。我使用的是postgres db。