java - 更新 GAE 中的对象

标签 java google-app-engine jdo datanucleus

我有一个无法解决的问题。我尝试在网上搜索解决方案,但没有找到任何通用的解决方案。我想更新数据存储中的一个对象,无论它是什么类。为此,这是我在该项目中使用的代码

我正在使用 DataNucleus 和 Google AppEngine。

这是我的 jdoconfig.xml

<?xml version="1.0" encoding="utf-8"?>
<jdoconfig xmlns="http://java.sun.com/xml/ns/jdo/jdoconfig"
   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
   xsi:noNamespaceSchemaLocation="http://java.sun.com/xml/ns/jdo/jdoconfig">

   <persistence-manager-factory name="transactions-optional">
       <property name="javax.jdo.PersistenceManagerFactoryClass"
           value="org.datanucleus.store.appengine.jdo.DatastoreJDOPersistenceManagerFactory"/>
       <property name="javax.jdo.option.ConnectionURL" value="appengine"/>
       <property name="javax.jdo.option.NontransactionalRead" value="true"/>
       <property name="javax.jdo.option.NontransactionalWrite" value="true"/>
       <property name="javax.jdo.option.RetainValues" value="true"/>
       <property name="datanucleus.appengine.autoCreateDatastoreTxns" value="true"/>
   </persistence-manager-factory>
</jdoconfig>

我的 PMF 类(class)

public final class PMF {
    private static final PersistenceManagerFactory pmfInstance =
        JDOHelper.getPersistenceManagerFactory("transactions-optional");

    private PMF() {}

    public static PersistenceManagerFactory get() {
        return pmfInstance;
    }

    public static PersistenceManager getPersistenceManager() {
        return pmfInstance.getPersistenceManager();
    }
}

我的 BaseModel 类,它是项目所有模型的父类(super class)

@Inheritance(strategy = InheritanceStrategy.SUBCLASS_TABLE)
@PersistenceCapable(detachable = "true", identityType = IdentityType.APPLICATION)
public abstract class BaseModel implements Serializable {

    @PrimaryKey
    @Persistent(valueStrategy = IdGeneratorStrategy.IDENTITY)
    protected Key id;

    public boolean equals(Object obj) {
        try {
            return obj instanceof BaseModel ? this.getId().getId() == ((BaseModel) obj).getId().getId() : false;
        } catch (Exception e) {
            return false;
        }
    }
}

这是我要保存的类(Project)

@PersistenceCapable(detachable = "true", identityType = IdentityType.APPLICATION)
public class Project extends BaseModel implements BeanModelTag {

    private static final long serialVersionUID = 3318013676594981107L;

    @Persistent
    private String name;

    @Persistent
    private String description;

    @Persistent
    private Date deadline;

    @Persistent
    private Integer status;

    @Persistent
    private Key manager;

    @Persistent
    private Set<Key> team;
}

为此,我尝试了几种方法。下面的方法成功保存了 Project 的新实例,但是当我调用更新已分离的对象时,只有字段 deadline 被更新,其他属性没有更新(但是,我进入 Debug模式以检查其他属性是否发生变化,是的,它们确实发生了变化,但只保存了最后期限)。

public void save(BaseModel object) {
    PersistenceManager pm = PMF.getPersistenceManager();
    try {
        pm.makePersistent(object);
    } finally {
        pm.close();
    }
}

所以,我尝试了以下代码

public void update(Project object) {
    PersistenceManager pm = PMF.get().getPersistenceManager();
    try {
        pm.currentTransaction().begin();

        Project p = pm.getObjectById(Project.class, object.getId());
        p.setName(object.getName());
        p.setDeadline(object.getDeadline());
        p.setDescription(object.getDescription());
        p.setTeam(p.getTeam());
        p.setStatus(object.getStatus());

        pm.currentTransaction().commit();
    } catch (Exception e) {
        e.printStackTrace();
        pm.currentTransaction().rollback();
    } finally {
        pm.close();
    }
}

而且它成功了。好的,它以这种方式工作,但我需要一个适用于所有模型的通用方法,所以我尝试了这个

public void update(BaseModel object) {
    PersistenceManager pm = PMF.get().getPersistenceManager();
    try {
        pm.currentTransaction().begin();

        BaseModel ob = pm.getObjectById(object.getClass(), object.getId());

        for (Field f : ob.getClass().getDeclaredFields()) {
            if (!f.toString().contains("final")) {
                f.setAccessible(true);
                for (Field g : object.getClass().getDeclaredFields()) {
                    g.setAccessible(true);
                    if (f.getName().equals(g.getName())) {
                        f.set(ob, g.get(object));
                    }
                    g.setAccessible(false);
                }
            }
            f.setAccessible(false);
        }

        pm.makePersistent(ob);
        pm.currentTransaction().commit();
    } catch (Exception e) {
        e.printStackTrace();
        pm.currentTransaction().rollback();
    } finally {
        pm.close();
    }
}

但它根本不起作用,没有保存任何内容,但是当我手动 System.out 属性时,它们被更改了。我尝试过使用和不使用 pm.makePersistent(ob); ,但没有成功。我不知道该怎么办。我在这个项目中有 120 个继承自 BaseModel 的模型,但我找不到一种方法来进行适用于我的模型的更新。

---------编辑------------

谢谢你的回答。这是我现在的解决方案。当然, printStrackTree 会离开那里。

public void update(BaseModel object) {
    PersistenceManager pm = PMF.get().getPersistenceManager();
    try {
        pm.currentTransaction().begin();

        BaseModel ob = pm.getObjectById(object.getClass(), object.getId());

        for (Field f : ob.getClass().getDeclaredFields()) {
            if (!Pattern.compile("\\bfinal\\b").matcher(f.toString()).find()) {
                f.setAccessible(true);
                for (Field g : object.getClass().getDeclaredFields()) {
                    g.setAccessible(true);
                    if (f.getName().equals(g.getName())) {
                        f.set(ob, g.get(object));
                        JDOHelper.makeDirty(ob, f.getName());
                    }
                    g.setAccessible(false);
                }
                f.setAccessible(false);
            }
        }

        pm.makePersistent(object);
        pm.currentTransaction().commit();
    } catch (Exception e) {
        e.printStackTrace();
        pm.currentTransaction().rollback();
    } finally {
        pm.close();
    }
}

最佳答案

使用反射更新字段不会被 JDO 字节码增强方法检测到。如果您想直接更新字段(无论是通过字段还是通过反射),那么您可以调用

JDOHelper.makeDirty(obj, "myFieldName");

完成更新后,JDO 将注册更改,从而在数据存储中更新。

关于java - 更新 GAE 中的对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7486710/

相关文章:

eclipse - 如何解决 xsi :noNamespaceSchemaLocation in jdoconfig. xml 上的验证错误

java - Camel 和 Google App Engine 没有 Spring?

node.js - 如何部署具有本地 npm 依赖项的 GAE 服务?

google-app-engine - 我应该如何在我的 Angular 2 项目中包含外部 .js 文件?

java - 如何使用 Apache ISI 中的 jdo 查询从表行号在 20 到 30 之间获取记录

java - Java 中子类和集合之间的类转换异常问题

java - 如何在ServletContextListener中获取spring ApplicationContext?

java - 从子树节点聚合值的算法

java - Gradle 可以将多个项目打包到一个 jar 中吗?

java - 数据仓库项目的存储过程与 JDO