私は GAE 1.7.0 w/ JDO(DataNucleus) を使用しています。コレクション属性を持つクラスを永続化すると、削除されたコレクション メンバーがデータストアから削除されません。切り離されたコピーからコレクション メンバーを削除します。新しいメンバーが正しく追加され、既存のメンバーが削除されないため、コレクションが増えるだけです。
@PersistenceCapable(identityType = IdentityType.APPLICATION, detachable="true")
public class Parent{
@PrimaryKey
@Persistent(valueStrategy = IdGeneratorStrategy.IDENTITY)
private Long id;
@Persistent(defaultFetchGroup="true", dependentElement="true")
private List<Child> children = new ArrayList<Child>(0);
}
@PersistenceCapable(identityType = IdentityType.APPLICATION, detachable="true")
public class Child implements Serializable {
@PrimaryKey
@Persistent(valueStrategy = IdGeneratorStrategy.IDENTITY)
@Extension(vendorName="datanucleus", key="gae.encoded-pk", value="true")
private String id;
@Persistent
@Extension(vendorName="datanucleus", key="gae.parent-pk", value="true")
private String parentId;
....
}
...
parent = pm.detachCopy(resultFromQuery);
// parent looks correct in dubugger: all children are fetched (and detached)
....
parent.getChildren().clear();
parent.getChildren().addAll(newChildren);
for (Child child: parent.getChildren())
child.setParentId(KeyFactory.createKeyString("Parent", parent.getId()));
// parent looks good in the debugger: old children were removed and new ones added
...
PersistenceManager pm = PMF.get().getPersistenceManager();
try {
pm.currentTransaction().begin();
pm.makePersistent(parent);
pm.currentTransaction().commit();
} catch (Exception e) {
} finally {
if (pm.currentTransaction().isActive())
pm.currentTransaction().rollback();
if (!pm.isClosed())
pm.close();
}
// problem in datastore: new children were created, old ones not removed
トランザクションで (オブジェクトをデタッチせずに) query-remove-persist を実行すると、問題が解決することに気付きました。これは一時的な回避策ですが、デタッチされたオブジェクトを更新したいと思います。