@GeneratedValue(strategy = GenerationType.AUTO)不能如愿以偿。

问题描述 投票:20回答:6

我试图将一个对象持久化到数据库中。 一直得到'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;
    }
hibernate jpa jboss derby
6个回答
22
投票

你可以使用 GenerationType.TABLE。这样一来,jpa使用序列表进行id分配,你可能永远不需要生成序列或自动增量值或触发器,降低了可移植性。

另外要注意的是,在java中int类型默认是以0发起的,所以你也可以把它去掉。


4
投票

在我的例子中,这是关于糟糕的方言。

hibernate.dialect=org.hibernate.dialect.H2Dialect

而不是:

hibernate.dialect=org.hibernate.dialect.PostgreSQL9Dialect

当我切换到生产数据库时,Hibernate试图使用策略。Hibernate试图使用为不同db引擎准备的策略。


2
投票

我也遇到过和你类似表现的问题。我最终发现我的数据库连接的配置是错误的:我连接的是一个旧的数据库,它的模式不正确。新的模式将主键列声明为

"ID" BIGINT NOT NULL GENERATED ALWAYS AS IDENTITY (START WITH 1, INCREMENT BY 1)

所以 数据库本身自动生成主键 而旧的模式则将其声明为

"ID" INTEGER NOT NULL

Hibernate为新的模式执行了正确的代码,而在旧的模式上却失败了,因为旧的模式要求SQL? INSERT 提供一个值 ID 列。


1
投票

当ID列是Int时,Hibernate会以沉默而神秘的方式失败。试着在代码中把它改成Long,在数据库中改成无符号64位整数。这为我解决了这个问题。


0
投票

或者尝试用 @GeneratedValue(strategy = GenerationType.AUTO)而不是 @GeneratedValue(strategy = GenerationType.SEQUENCE).


0
投票

以防有人到了这个帖子里和我遇到同样的问题。我一直在努力解决这个问题,现在已经有几个小时了,我只想把我的解决方案贴出来。似乎ID总是被设置为0,这引发了一个错误。

模型中的id属性必须是 "long "或 "Integer "类型,而不是 "long "或 "integer"(注意大写)。我放的是 "long",这是一个基元类型,不能为空。因此默认值是0,Hibernate没有意识到这是一个新的实体,于是试图用ID=0保存它,这显然是行不通的。

将其改成可以为空的引用类型 "Long",就解决了这个问题。


0
投票

我也在为同样的问题而苦恼,不知何故,对我来说,long类型比long类型更有效:)。作为一个变通办法,我把数据库中的id列转换为SERIEL。我使用的是postgres db。

© www.soinside.com 2019 - 2024. All rights reserved.