【发布时间】:2012-11-21 20:56:03
【问题描述】:
在 Glassfish AS 上使用 DataNucleus 作为 ORM 框架时遇到问题。实际上我正在尝试使用连接到 MongoDb。我预计这里会出现问题,因为每个人都知道 nosql 数据库不是标准的一部分,但是我在使用 DataNucleus 时遇到了问题。那是我没想到的。我对 DataNucleus 和 MongoDB 都不熟悉,很遗憾在 WEB 上找不到任何相关信息,所以我决定一步一步来。
我创建了一个使用 DataNucleus 和 MongoDB 并使用 Maven 构建的小型命令行应用程序。相关代码如下
persistece.xml
<?xml version="1.0" encoding="UTF-8"?>
<persistence version="1.0"
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_1_0.xsd">
<persistence-unit name="just_test">
<provider>org.datanucleus.api.jpa.PersistenceProviderImpl</provider>
<properties>
<property name="datanucleus.ConnectionURL" value="mongodb:/justtest"/>
<property name="datanucleus.autoCreateSchema" value="true"/>
<property name="datanucleus.validateTables" value="false"/>
<property name="datanucleus.validateConstraints" value="false"/>
</properties>
</persistence-unit>
</persistence>
持久化实体的代码
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.Persistence;
public class App {
public static void main(String[] args) {
EntityManagerFactory emf = Persistence.createEntityManagerFactory("just_test");
// Create entity manager
EntityManager em = null;
String id;
try {
em = emf.createEntityManager();
EntityTransaction tx = em.getTransaction();
TestEntity testEntity = new TestEntity();
tx.begin();
em.persist(testEntity);
em.flush();
tx.commit();
id = testEntity.getId();
} finally {
if (null != em) {
em.close();
}
}
try {
em = emf.createEntityManager();
TestEntity testEntity = em.find(TestEntity.class, id);
System.out.println(testEntity);
} finally {
if (null != em) {
em.close();
}
}
}
pom.xml
...
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>3.8.1</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.mongodb</groupId>
<artifactId>mongo-java-driver</artifactId>
<version>[2.9.3, )</version>
<!--<scope>provided</scope>-->
</dependency>
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-core</artifactId>
<version>[2.9, )</version>
</dependency>
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-mongodb</artifactId>
<version>[2.9, )</version>
</dependency>
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-api-jpa</artifactId>
<version>[3.0, )</version>
</dependency>
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-api-jdo</artifactId>
<version>[3.0, )</version>
</dependency>
<dependency>
<groupId>org.apache.geronimo.specs</groupId>
<artifactId>geronimo-jpa_2.0_spec</artifactId>
<version>[1.0, )</version>
</dependency>
<dependency>
<groupId>javax.jdo</groupId>
<artifactId>jdo2-api</artifactId>
<version>2.2</version>
</dependency>
<dependency>
<groupId>javax.transaction</groupId>
<artifactId>transaction-api</artifactId>
<version>[1.1-rev-1, )</version>
</dependency>
<dependency>
<groupId>javax.persistence</groupId>
<artifactId>persistence-api</artifactId>
<version>1.0b</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.datanucleus</groupId>
<artifactId>maven-datanucleus-plugin</artifactId>
<version>3.1.0-m3</version>
<configuration>
<log4jConfiguration>${basedir}/log4j.properties</log4jConfiguration>
<verbose>true</verbose>
</configuration>
<executions>
<execution>
<phase>process-classes</phase>
<goals>
<goal>enhance</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
...
此代码运行良好。
一旦我尝试部署一个使用相同代码的应用程序,首先发生的事情是 GF 无法找到某些类。这些类虽然在一个 .ear 文件中。我不得不将几个类放到 GF_HOME/glassfish/domains//lib 目录中,让 GF 能够找到这些类,但最后我以异常结束了
Caused by: javax.persistence.PersistenceException: No Persistence provider for EntityManager named just_test
值得一提的是,org.datanucleus.api.jpa.PersistenceProviderImpl 类在我也放在域的 lib 目录中的 datanucleus-api-jpa 文件中。
我也尝试使用 CDI 进行如下操作
@PersistenceContext(unitName = "just_test")
protected EntityManager entityManager;
@Override
public void persist(TestEntity testEntity) {
entityManager.persist(testEntity);
}
这个方法也没有成功。我有一个例外
java.lang.ClassCastException: Cannot cast org.datanucleus.api.jpa.PersistenceProviderImpl to javax.persistence.spi.PersistenceProvider
任何想法或文章和/或文档的链接将不胜感激。谢谢
下面是我尝试在 Glassfish 上部署的应用程序的依赖项列表。我必须将所有“提供”范围的依赖项放入 Glassfish 的 lib 目录中
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-core</artifactId>
<version>[2.9, )</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-mongodb</artifactId>
<version>[2.9, )</version>
</dependency>
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-api-jpa</artifactId>
<version>[3.0, )</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.apache.geronimo.specs</groupId>
<artifactId>geronimo-jpa_2.0_spec</artifactId>
<version>[1.0, )</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.datanucleus</groupId>
<artifactId>datanucleus-api-jdo</artifactId>
<version>[3.0, )</version>
</dependency>
<dependency>
<groupId>javax.jdo</groupId>
<artifactId>jdo2-api</artifactId>
<version>2.2</version>
</dependency>
【问题讨论】:
-
您的 CLASSPATH 中似乎有 JPA1 jar (javax.persistence) 和 JPA2 jar (geronimo-jpa_2.0_spec)。如果我是你,我会先决定要使用哪个版本的 JPA。显然那里使用的 DN 版本完全实现了 JPA2
-
@DataNucleus 感谢您的评论。之所以存在这种依赖关系,是因为我首先找到了 JPA1 的示例,然后修改了代码以使用 JPA2。我删除了不必要的依赖,但没有帮助。我有相同的堆栈跟踪。我还修改了我的帖子并将我用于 GF 的整个依赖项列表。
-
当 DN 版本实现 JDO 3.0 时,出于某种原因,您拥有 JDO 2.2。 Glassfish 将在某处拥有自己的 JPA.jar,因此它可能不想要另一个 JPA.jar 或感到困惑。事实是“org.datanucleus.api.jpa.PersistenceProviderImpl”确实完全实现了 javax.persistence.spi.PersistenceProvider 所以它是一个类加载的东西
-
看来是类加载问题。问题是为什么。
-
例如,我不需要采取任何特殊步骤来使用 Hibernate。你能推荐任何与之相关的文件吗?我在这里查看datanucleus.org/products/accessplatform_2_2/guides/jpa/…,但还没有找到解决方案。