1

たぶん、この質問は答えられたかもしれませんが、そうではないかもしれません。

楽観的なロックでオブジェクトを更新したい

...
EntityManager.merge(...);
EntityManager.flush(...);
...

私のコードは正常に動作します。私が欲しいのは、オブジェクトのバージョン番号を更新し、オブジェクト参照を保持することです。

長い間グーグルで検索しましたが、答えが見つかりませんでした。

私の最後の手段は、nativeQueryを作成し、バージョン番号を取得することです。

テストコードは次のとおりです。

    EntityManagerFactory factory = Persistence.createEntityManagerFactory(DBManager.getUnidadPersistencia());
    EntityManager emA = factory.createEntityManager();

    Grupo g = emA.find(Grupo.class, 1L);

    emA.close();

    //In some place 
    g.getSolicitanteList().get(0).setObservaciones("ZZZZ"); //Originally XXXX

    //...


    //Then I have to save the changes
    EntityManager emB = factory.createEntityManager();
    EntityTransaction tx = emB.getTransaction();

    try {
        tx.begin();
        emB.merge(g);
        emB.flush();
        tx.commit();
        //Here comes exception
        emB.refresh(g);
    } catch (Exception e) {
        if (tx.isActive()) {
            tx.rollback();
        }
        logger.log(Level.SEVERE, null, e);
    }
    emB.close();

そしてスタックトレース:

30/07/2012 05:15:31 PM org.rp.co5parser.TestGrupo refreshTest
GRAVE: null
java.lang.IllegalArgumentException: Can not refresh not managed object: org.rp.co5parser.alojamiento.Grupo[ idGrupo=1 ].
at org.eclipse.persistence.internal.jpa.EntityManagerImpl.refresh(EntityManagerImpl.java:943)
at org.eclipse.persistence.internal.jpa.EntityManagerImpl.refresh(EntityManagerImpl.java:849)
at org.rp.co5parser.TestGrupo.refreshTest(TestGrupo.java:89)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:45)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:42)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:263)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:68)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:47)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:231)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:60)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:229)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:50)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:222)
at org.junit.runners.ParentRunner.run(ParentRunner.java:300)
at org.apache.tools.ant.taskdefs.optional.junit.JUnit4TestMethodAdapter.run(JUnit4TestMethodAdapter.java:109)
at org.apache.tools.ant.taskdefs.optional.junit.JUnitTestRunner.run(JUnitTestRunner.java:520)
at org.apache.tools.ant.taskdefs.optional.junit.JUnitTestRunner.launch(JUnitTestRunner.java:1060)
at org.apache.tools.ant.taskdefs.optional.junit.JUnitTestRunner.main(JUnitTestRunner.java:911)

それは大丈夫ですか?

前もって感謝します....

4

1 に答える 1

3

グーグルする前に、ドキュメントを読んでください:

void refresh(java.lang.Object entity)

データベースからインスタンスの状態を更新し、エンティティに加えられた変更がある場合はそれを上書きします。

編集:

発生するエラーはOptimisticLockExceptionではありません。これはIllegalArgumentExceptionであり、非常に明確なエラーメッセージが表示されます。管理されていないオブジェクトを更新できません。管理されているエンティティのみを更新できます。そして、渡すオブジェクトは分離オブジェクトです。

の結果を無視しなければ、エンティティを更新する必要はまったくありませんmerge()merge()デタッチされたオブジェクトを取得し、その状態を同じIDの管理対象(接続済み)オブジェクトにコピーして、管理対象オブジェクトを返します。したがって、次のことを行う必要があります。

try {
    tx.begin();
    g = emB.merge(g); // assign the merged, attached object, having the new version to g
    tx.commit(); // no need to flush. commit flushes automatically
} catch (Exception e) {
    if (tx.isActive()) {
        tx.rollback();
    }
    logger.log(Level.SEVERE, null, e);
}
emB.close();
于 2012-07-30T20:54:13.477 に答える