Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/java/307.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java EntityManager.persist()可以工作,但EntityManager.merge()不能_Java_Merge_Persistence_Eclipselink_Entitymanager - Fatal编程技术网

Java EntityManager.persist()可以工作,但EntityManager.merge()不能

Java EntityManager.persist()可以工作,但EntityManager.merge()不能,java,merge,persistence,eclipselink,entitymanager,Java,Merge,Persistence,Eclipselink,Entitymanager,我们正在使用EclipseLink 2.5.2和WildFly 9.0.0 下面是一个方法,它每小时被EJB计时器指定的特定次数调用一次。出于测试目的,我已将其更改为每分钟触发一次,在我的特定配置中,此方法将每分钟调用10次,因为需要进行10次更新: @Stateless public class NetCentricDataService { @PersistenceContext(unitName = "NetCentricPersistenceUnit") private

我们正在使用EclipseLink 2.5.2和WildFly 9.0.0

下面是一个方法,它每小时被EJB计时器指定的特定次数调用一次。出于测试目的,我已将其更改为每分钟触发一次,在我的特定配置中,此方法将每分钟调用10次,因为需要进行10次更新:

@Stateless
public class NetCentricDataService {

    @PersistenceContext(unitName = "NetCentricPersistenceUnit")
    private EntityManager em;

    public void insertUpdate(A a, B b, boolean insert) {
        if (insert) {
            em.persist(a);
            em.flush();
            if (b != null) {
                em.persist(b);
                em.flush();
            }
        } else {
            em.merge(a);
            if (b != null) {
                em.merge(b);
            }
            em.flush();
        }
    }
}
当我清除与下面两个实体类关联的表时,ifinsert块执行,persist调用工作正常,数据库更新;但是,如果数据库中存在条目,则执行else块,合并调用运行时不会出现异常,但数据库不会更新。在方法调用完成后,我将在SQL Developer中检查这一点

编辑:例如,以下操作不起作用:

A a = dataService.getA(); // gets exisiting A from DB
a.setSubject("My Subject");
B b = dataService.getB(); // gets existing B from DB
b.setFoo(123.45);
insertUpdate(a, b, false);
以下是实体类别A:

@Entity
@Table(name = "A")
@SuppressWarnings("unused")
public class A implements Serializable {

    /**
     * Method called by container before DB insert.
     */
    @PrePersist
    public void beforeInsert() {
        if (objectType == null) {
            objectType = ObjectTypeEnum.UNKNOWN;
        }
    }

    @Id
    @Column(name = "MY_UID")
    private String myUid;

    @Column(name = "MY_NUMBER")
    private int myNumber;

    @Column(name = "SUBJECT")
    private String subject;

    @Enumerated(EnumType.ORDINAL)
    @Column(name = "OBJECT_TYPE")
    private ObjectTypeEnum objectType;

    // other columns, getters, setters, equals, etc.

}
这是实体类别B:

@Entity
@Table(name = "B")
@SuppressWarnings("unused")
public class B implements Serializable {

    @Id
    @Column(name = "MY_UID", nullable = false)
    private String myUid;

    @Column(name = "FOO")
    private double foo;

    @Column(name = "BAR")
    private double bar;
}
注意:B中的MY_UID是a中MY_UID的外键

注意:在我试图更新的任何列上都没有updateable=false属性

让我知道是否有任何其他文件/方法/等,我可以包括或如果有任何更多的信息,我可以提供。值得注意的是,我没有维护我发布的任何代码,但我最近一直在使用它。我问了维修人员,他说他不知道为什么会发生这种情况。提前感谢您的帮助

编辑-以下是standalone-full.xml中的数据源配置:

<subsystem xmlns="urn:jboss:domain:datasources:3.0">
        <datasources>
            <datasource jndi-name="java:jboss/datasources/ExampleDS" pool-name="ExampleDS" enabled="true" use-java-context="true">
                <connection-url>jdbc:h2:mem:test;DB_CLOSE_DELAY=-1;DB_CLOSE_ON_EXIT=FALSE</connection-url>
                <driver>h2</driver>
                <security>
                    <user-name>sa</user-name>
                    <password>****</password>
                </security>
            </datasource>
            <datasource jta="true" jndi-name="java:jboss/datasources/SFDS" pool-name="SFDS" enabled="true" use-java-context="true" use-ccm="true">
                <connection-url>jdbc:oracle:thin:@DB:1521:ORACLE_SID</connection-url>
                <driver>ojdbc6</driver>
                <new-connection-sql>select 1 from dual</new-connection-sql>
                <pool>
                    <min-pool-size>10</min-pool-size>
                    <max-pool-size>100</max-pool-size>
                    <prefill>false</prefill>
                    <use-strict-min>false</use-strict-min>
                    <flush-strategy>FailingConnectionOnly</flush-strategy>
                </pool>
                <security>
                    <user-name>wildfly_user</user-name>
                    <password>****</password>
                </security>
                <validation>
                    <check-valid-connection-sql>select 1 from dual</check-valid-connection-sql>
                    <use-fast-fail>true</use-fast-fail>
                </validation>
            </datasource>
            <drivers>
                <driver name="h2" module="com.h2database.h2">
                    <xa-datasource-class>org.h2.jdbcx.JdbcDataSource</xa-datasource-class>
                </driver>
                <driver name="ojdbc6" module="com.oracle.jdbc">
                    <xa-datasource-class>oracle.jdbc.OracleDriver</xa-datasource-class>
                </driver>
            </drivers>
        </datasources>
    </subsystem>

问题似乎在于,在尝试调用merge之前,我们调用了一个带有只读提示的SELECT查询。删除此提示将允许合并成功显示在数据库中。谢谢克里斯的帮助

在您的示例中,如何设置ID?您是否正在尝试使用merge插入ID为空的A和B实例?如果B的ID是a的外键,它是如何设置的?对不起,我在合并之前编辑了我的示例。它从已经设置了MY_UID字段的数据库中获取现有的A和B。例如,假设表是空的。对程序的第一次调用将创建一个新的a和B,然后调用a.setMyUidUUID.randomUUID.toString;,b、 setMyUida.getMyUid;,以及传递的任何更新。现在他们已经准备好了。然后下一个调用将获得现有的A和B,设置一些新字段,然后尝试合并。如果您使用容器的连接池。数据源使用什么数据库和jdbc驱动程序。这取决于您如何运行WildFly,通常standalone.xml是您感兴趣的文件。我在问题中包括了数据源配置。查看我的编辑。getA是什么样子的?我所能看到的唯一一件事情是,如果您想知道如何在共享缓存中更改对象,那么当合并更改的对象时,Eclipselink不会看到有更改。请确保对A和B的查询没有试图获取只读实例,也没有进行可能导致其从共享缓存返回只读实例的优化。尝试在合并之前显式调用em.find,并将find返回的内容与实体中的内容进行比较。