0

我将嵌入式 glassfish (3.1.2.2) 与 junit (4.11) 和 JDK 1.7 一起使用,尽管我的源和目标设置为 1.6(maven-compiler-plugin 配置)。

以下是我的代码:

人.java

@Entity
public class Person implements Serializable {

    private static final long serialVersionUID = 81398385247591972L;

    @Id
    @GeneratedValue
    private Long id;
    @Version
    private Long version;
    @Column(length = 15, nullable = false, unique = true, updatable = false)
    private String username;
    @Column(length = 50)
    private String status;

    // Constructors

    // getters/setters

    // hashCode, equals, toString

}

服务.java

@Stateless
public class Service {

    @PersistenceContext(unitName = "ExamplePU", type = PersistenceContextType.TRANSACTION)
    private EntityManager em;

    public Person add(Person person) {
        em.persist(person);
        return person;
    }

    public Person find(Long id) {
        return em.find(Person.class, id);
    }

    public Person modify(Person person) {
        return em.merge(person);
    }

    // some more code ...

}

服务测试.java

public class ServiceTest {

    private static EJBContainer ejbContainer;
    private static Service service;

    // @BeforeClass, @AfterClass, @Before, @After

    @Test
    public void testMerge() {
        Person person;

        /* Step 1 */person = service.add(new Person("username", "status"));
        print("Added : " + person);

        person.setStatus("Away");
        /* Step 2 */person = service.modify(person);
        print("Merged (status change) : " + person);

        person.setUsername("UsErNaMe");
        /* Step 3 */person = service.modify(person);
        print("Merged (username change) : " + person);
    }

    // Some more tests

}

第 1 步生成以下 SQL(如预期的那样):

INSERT INTO PERSON (ID, STATUS, USERNAME, VERSION) VALUES (?, ?, ?, ?)
    bind => [1, status, username, 1]

第 2 步生成以下 SQL(如预期的那样):

UPDATE PERSON SET STATUS = ?, VERSION = ? WHERE ((ID = ?) AND (VERSION = ?))
    bind => [Away, 2, 1, 1]

第 3 步不会生成任何 SQL,但不会引发任何异常,这是我所期待的,因为“用户名”被注释为@Column(..., updatable = false)。print(...) 方法打印以下输出:

合并(用户名更改):人员 [id=1,版本=2,用户名=UserNaMe,状态=Away]

这次 merge() 操作更新了用户名,但没有更新版本。此外,现在数据库与 EntityManager 缓存不同步。

这是预期的,还是 EclipseLink 中的错误?

更新

预期结果是上述步骤 3 中的异常。

更新

已在此处提交错误。

4

1 回答 1

1

您将该列标记为不可更新,并且 EclipseLink 检测到对您告诉它合并的人所做的唯一更改是用户名。但不得更新用户名。所以它不会发出任何 SQL 更新查询。

如果将列标记为不可更新,则不应更新它。

因此,为了清楚起见,您观察到的行为是预期的行为。

于 2013-02-27T09:15:21.663 回答