EclipseLink 的 @UuidGenerator 导致 NullPointerException

Posted

技术标签:

【中文标题】EclipseLink 的 @UuidGenerator 导致 NullPointerException【英文标题】:EclipseLink's @UuidGenerator leads to NullPointerException 【发布时间】:2014-12-07 22:48:19 【问题描述】:

我尝试开始使用 EclipseLink 的 @UuidGenerator 注释来生成 ID,但由于 NPE 失败。

我正在使用 Java EE 6 + GlassFish 4.1。实体是从 servlet 创建的。 EclipseLink 版本是 2.6.0-M3。简单的长 id "@GeneratedValue(strategy = GenerationType.AUTO)" 工作正常。

实体代码:

@Entity
@UuidGenerator(name="QUANTITY_GEN")
public class Quantity implements Serializable 

    private static final long serialVersionUID = 1L;
    @Id
    @GeneratedValue(generator = "QUANTITY_GEN")
    private String id;

...


Servlet 代码:

@WebServlet(name = "Qtest", urlPatterns =

    "/Qtest"
)
public class Qtest extends HttpServlet

    @EJB
    private QuantityFacade quantityFacade;

    protected void processRequest(HttpServletRequest request, HttpServletResponse response)
        throws ServletException, IOException
    
        ...

        Quantity qt = new Quantity();
        quantityFacade.create(qt);

        ...
    

追踪:

Caused by: java.lang.NullPointerException
    at org.eclipse.persistence.internal.sequencing.SequencingManager.getNextValue(SequencingManager.java:1067)
    at org.eclipse.persistence.internal.sequencing.ClientSessionSequencing.getNextValue(ClientSessionSequencing.java:70)
    at org.eclipse.persistence.internal.descriptors.ObjectBuilder.assignSequenceNumber(ObjectBuilder.java:361)
    at org.eclipse.persistence.internal.descriptors.ObjectBuilder.assignSequenceNumber(ObjectBuilder.java:320)
    at org.eclipse.persistence.internal.sessions.UnitOfWorkImpl.assignSequenceNumber(UnitOfWorkImpl.java:486)
    at org.eclipse.persistence.internal.sessions.UnitOfWorkImpl.registerNotRegisteredNewObjectForPersist(UnitOfWorkImpl.java:4290)
    at org.eclipse.persistence.internal.sessions.RepeatableWriteUnitOfWork.registerNotRegisteredNewObjectForPersist(RepeatableWriteUnitOfWork.java:518)
    at org.eclipse.persistence.internal.sessions.UnitOfWorkImpl.registerNewObjectForPersist(UnitOfWorkImpl.java:4235)
    at org.eclipse.persistence.internal.jpa.EntityManagerImpl.persist(EntityManagerImpl.java:496)
    at com.sun.enterprise.container.common.impl.EntityManagerWrapper.persist(EntityManagerWrapper.java:287)
    at uuidtest.AbstractFacade.create(AbstractFacade.java:24)

请告知可能有一些明显的错误?

【问题讨论】:

实际上这个@UuidGenerator 似乎需求量不大。它的文档记录很差,并且只生成字符串表示,而不能使用 UUID。 【参考方案1】:

看来这确实是 EclipseLink 的错误,而且很难很快修复,所以我使用了一种变通方法 - 直接 id 生成,如此处所述http://blog.xebia.com/2009/06/03/jpa-implementation-patterns-using-uuids-as-primary-keys/

@Id
private String id;

public AbstractBaseEntity() 
    this.id = UUID.randomUUID().toString();

【讨论】:

【参考方案2】:

我使用以下代码来解决这个问题。在使用 BeanUtils.copyProperties() 时,使用默认构造函数给我带来了使用复合键克隆实体的问题。因此我使用了@PrePersist。这是在第一次在数据库中创建行之前调用的

@MappedSuperclass
@Getter
@Setter
public abstract class AbstractMasterUUIDEntity extends AbstractMasterEntity<String> 

    @Id
    private String id;

    @PrePersist
    public void generateId() 
        if(this.id == null ) 
            this.id = UUID.randomUUID().toString();
        
    


【讨论】:

以上是关于EclipseLink 的 @UuidGenerator 导致 NullPointerException的主要内容,如果未能解决你的问题,请参考以下文章

EclipseLink 错误?简单的原生 SQL 查询异常

Eclipselink 2.5 使用 Maven 生成元模型

Eclipselink 实体类自动扫描发现非实体

eclipselink.weaving 属性的值和区别是啥?

eclipselink 连接池

Eclipselink 实体持续存在问题