JPA 与 OSGi 示例在 serviceReference 中返回 null

Posted

技术标签:

【中文标题】JPA 与 OSGi 示例在 serviceReference 中返回 null【英文标题】:JPA with OSGi example returning null in serviceReference 【发布时间】:2013-12-07 07:31:51 【问题描述】:

我一直在研究 JPA 将与 OSGi 一起使用的演示示例。

问题是,我可以在捆绑后启动/停止服务,但是我无法获得serviceReference。因此,我的 JPA 实现没有被执行。

以下是代码:

Activator.java

package manning.osgi.jpa;

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;

import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceReference;

public class Activator implements BundleActivator 

public void start(BundleContext context) throws Exception 
    LoginEvent loginEvent = new LoginEvent();

    // Set login event...
    loginEvent.setUserid("alex");

    try 
        ServiceReference [] serviceReferences =
                context.getServiceReferences(
                        EntityManagerFactory.class.toString(),
                        "(osgi.unit.name=LoginEvent)");

        System.out.println("Service References Created");

         if (serviceReferences != null) 
             EntityManagerFactory emf = 
                 (EntityManagerFactory) context.getService(serviceReferences[0]);

             EntityManager em = emf.createEntityManager();

             persistLoginEvent(em, loginEvent);
             System.out.println("Transaction started");

             loginEvent = retrieveLoginEvent(em, loginEvent.getId());
             System.out.println("Transaction completed");

             em.close();
             emf.close();
         

     catch (Exception e) 
        System.out.println("Exception"+e.getMessage());
    



private void persistLoginEvent(EntityManager em, LoginEvent loginEvent) 
    em.getTransaction().begin();

    em.persist(loginEvent);

    em.getTransaction().commit();


private LoginEvent retrieveLoginEvent(EntityManager em, int id) 
    em.getTransaction().begin();

    LoginEvent loginEvent = em.find(LoginEvent.class, id);
    loginEvent.getUserid();
    loginEvent.getId();
    loginEvent.getTimestamp();

    em.getTransaction().commit();

    return loginEvent;


public void stop(BundleContext context) throws Exception 

LoginEvent.java

package manning.osgi.jpa;

import javax.persistence.*;

@Entity
public class LoginEvent 

@Id
@GeneratedValue
private int id;
private String userid;
private long timestamp;

public int getId() 
    System.out.println("\nID: "+this.id);
    return this.id;

public String getUserid() 
    System.out.println("\nUser ID: "+this.userid);
    return this.userid;     

public void setUserid(String userid) 
    this.userid = userid;

public long getTimestamp() 
    System.out.println("\nTime Stamp: "+this.timestamp);
    return this.timestamp;

public void setTimestamp(long timestamp) 
    this.timestamp = timestamp;


persistence.xml

<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="LoginEvent">
    <class>manning.osgi.jpa.LoginEvent</class>        
      <properties>
        <property name="javax.persistence.jdbc.driver"
            value="org.apache.derby.jdbc.EmbeddedDriver"/>
        <property name="javax.persistence.jdbc.url" 
            value="jdbc:derby:derbyDB;create=true"/>
    </properties>
</persistence-unit>
</persistence>

MANIFEST.MF

Manifest-Version: 1.0
Bundle-ManifestVersion: 2
Bundle-Name: OsgiDemo
Bundle-SymbolicName: manning.osgi.jpa
Bundle-Version: 1.0.0.qualifier
Bundle-Activator: manning.osgi.jpa.Activator
Bundle-RequiredExecutionEnvironment: JavaSE-1.6
Import-Package: javax.persistence,
 org.osgi.framework;version="1.3.0"
Bundle-ActivationPolicy: lazy
Require-Bundle: javax.persistence;bundle-version="2.1.0"

代码正在编译,但由于serviceReference 为空,我无法继续。我重新检查了persistence-unit 名称及其正确性。

任何人都可以帮助我解决我在这里缺少的东西。 谢谢。

【问题讨论】:

【参考方案1】:

在您的 Activator 中,您无法确定您需要的 ServiceReference 是否已经可用。您的捆绑包可能会在注册该服务的捆绑包之前启动。也有可能是其他bundle无法启动,所以服务不会被注册。

如果你想写很多代码,你可以在你的 start 函数中创建一个 ServiceTracker 并在跟踪器的 addingService 函数中执行相同的操作,就像你在 BundleActivator 的 start 函数中所做的那样。

如果您想减少工作量,请使用声明式服务(网上有很多教程)。在那里你可以定义你的组件在你需要的 OSGi 服务可用之前不应该启动。

顺便说一句:如果你使用了getService(serviceReference)函数,你应该调用unGetService

【讨论】:

我做 OSGi 已经快一年了。忘了ServiceTracker。我想这可能是问题所在。是的,为 DS +1。 是的,您可能是正确的,在我注册之前没有开始服务。但是我似乎无法以某种方式修复它。我了解到en.wikipedia.org/wiki/OSGi_Specification_Implementations 说 JPA 在 Aries 或 Gemini 中实现。我一直在使用 Eclipse 和 Equinox 作为我的 JPA 示例。 白羊座和双子座没有实现 JPA。他们实现了一个扩展器,该扩展器使用 persistence.xml 获取捆绑包,并基于 JPA 引擎创建新的 EntityManagerFactory OSGi 服务。更具体地说,它们实现了 OSGi 企业规范的“Java Persistence API”章节。如果您拥有所需的所有 aries 捆绑包,您应该会看到 EntityManagerFactory 服务。您如何使用该服务与 JPA 无关,而是更多 OSGi 的东西。例如。您可以使用 DS 或 Blueprint 将其注入到您的课程中。

以上是关于JPA 与 OSGi 示例在 serviceReference 中返回 null的主要内容,如果未能解决你的问题,请参考以下文章

OSGI、Servlets 和 JPA hello world / 教程 / 示例

Flyway 与 JPA + OSGi 的集成

在有或没有 OSGi 的情况下使用 JPA 的区别

在 OSGI (Karaf) 中使用 EclipseLink JPA

使用 Hibernate Unmanaged JPA 在 Equinox OSGI 框架中获取 EntityManagerFactory

未注入 OSGI 服务 JPA PersistenceContext