在学习过程中,Maven创建了一个简单的JPA项目(带有EclipseLink实现2.0.2的Java Persistence 1.0.2)。这是一个Application Managed环境。所以我手动控制EntityManager的生命周期。
主源代码使用的persistence.xml文件与单元测试代码使用的文件不同(也建议使用here)。主代码使用Oracle DB,测试代码使用内存中的Derby。运行单元测试正在更新Oracle DB(!),我最终设法通过在XML文件中使用两个不同的持久性单元来解决这个问题。
但是,我不明白为什么要解决这个问题。我手动创建并关闭实体管理器,它们不是同时运行的。我很确定Maven(或者我设置它的方式)不会弄乱资源(XML文件)。事实上,通过查看Maven的调试输出,我可以看到它正在使用正确的XML文件进行单元测试。事实上,我不明白为什么这应该是一个问题开始。
有人可以开导我吗?
- 更新
这是src / main / resources / META-INF / Persistence.xml:
<?xml version="1.0" encoding="UTF-8"?>
<persistence xmlns="http://java.sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"
version="2.0">
<persistence-unit name="MainPU" transaction-type="RESOURCE_LOCAL">
<provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
<class>org.domain.Book</class>
<class>org.domain.Tag</class>
<properties>
<property name="eclipselink.target-database" value="Oracle" />
<property name="eclipselink.logging.level" value="INFO" />
<property name="javax.persistence.jdbc.driver" value="oracle.jdbc.OracleDriver" />
<property name="javax.persistence.jdbc.url" value="jdbc:oracle:thin:@xxx:1526:XE" />
<property name="javax.persistence.jdbc.user" value="usr" />
<property name="javax.persistence.jdbc.password" value="pass" />
</properties>
</persistence-unit>
</persistence>
这里是src / test / resources / META-INF / Persistence.xml:
<?xml version="1.0" encoding="UTF-8"?>
<persistence xmlns="http://java.sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"
version="2.0">
<persistence-unit name="TestPU" transaction-type="RESOURCE_LOCAL">
<provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
<class>org.domain.Book</class>
<class>org.domain.Tag</class>
<properties>
<property name="eclipselink.target-database" value="DERBY" />
<property name="eclipselink.ddl-generation" value="drop-and-create- tables" />
<property name="eclipselink.logging.level" value="FINE" />
<property name="javax.persistence.jdbc.driver" value="org.apache.derby.jdbc.EmbeddedDriver" />
<property name="javax.persistence.jdbc.url" value="jdbc:derby:memory:testDB;create=true" />
<property name="javax.persistence.jdbc.user" value="usr" />
<property name="javax.persistence.jdbc.password" value="pass" />
</properties>
</persistence-unit>
</persistence>
主要源代码:
....
EntityManagerFactory emf = Persistence.createEntityManagerFactory("MainPU");
EntityManager em = emf.createEntityManager();
EntityTransaction tx = em.getTransaction();
tx.begin();
Iterator<Book> booksItr = books.iterator();
while (booksItr.hasNext())
em.persist(booksItr.next());
tx.commit();
em.close();
emf.close();
...
和单元测试源代码几乎相同,只是它使用“TestPU”来创建EntityManagerFactory。
如果您有2个不同的数据库,则需要2个不同的persistence.xml文件。在测试persistence.xml文件中,如果有一个测试类,则只需要一个持久性单元。当您编写第二个测试类来测试访问内存中Derby数据库的业务规则时,您需要在测试persistence-unit
中进一步使用persistence.xml
。我认为您的测试代码访问Oracle DB的原因是您在test / persistence.xml文件中定义了Oracle DB的持久性单元。你应该删除它,你应该只有derby内存数据库的持久性单位。