2

RESTful Web サービスをホストする組み込みの Jetty Web サーバーを実行する Java SE アプリケーションがあります。Java SE 部分と Jetty で実行されている Web アプリケーションの両方で Spring を使用したいと考えています。一部の Bean (dao やエンティティ マネージャーなど) は、2 つの部分で共有する必要があります。アプリケーションを起動すると、Bean とエンティティ マネージャーの両方のバージョンが重複しています。

WARN EntityManagerFactoryRegistry.addEntityManagerFactory:80 - HHH000436: Entity manager factory name (persistenceUnit) is already registered.  If entity manager will be clustered or passivated, specify a unique value for property 'hibernate.ejb.entitymanager_factory_name'

Java SE の部分では、最初にアプリケーション コンテキストを作成します。

ApplicationContext context = new ClassPathXmlApplicationContext("classpath:server-context.xml","classpath:application-context.xml");

Web アプリケーションでは、Spring は web.xml で構成されます。

<servlet>
    <servlet-name>Jersey REST Service</servlet-name>
    <servlet-class>com.sun.jersey.spi.spring.container.servlet.SpringServlet</servlet-class>
    <init-param>
        <param-name>com.sun.jersey.spi.container.ResourceFilters</param-name>
        <param-value>example.restful.server.filter.SecurityFilterFactory</param-value>
    </init-param>
    <init-param>
        <param-name>com.sun.jersey.config.property.resourceConfigClass</param-name>
        <param-value>com.sun.jersey.api.core.PackagesResourceConfig</param-value>
    </init-param>
    <init-param>
        <param-name>com.sun.jersey.config.property.packages</param-name>
        <param-value>example.restful.rest</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
</servlet>
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:application-context.xml</param-value>
</context-param>
<listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

すべての Bean を複製する代わりに、Web アプリケーションで既に作成されているアプリケーション コンテキストを使用するにはどうすればよいですか?

4

2 に答える 2

4

ContextLoaderListener独自のオーバーライドcreateWebApplicationContextを作成して、JavaSE アプリの ApplicationContext を返すことができると思います(何らかの種類の ApplicationContextHolder があり、静的アクセスを保持していると仮定します)。

WebApplicationContext簡単に思えますが、これを行うには a ではなく aを返す必要がありますClassPathXmlApplicationContext。次のようなことができると思います。

public class MyContextLoaderListener extends ContextLoaderListener{
    protected WebApplicationContext createWebApplicationContext(ServletContext sc,
                                                        ApplicationContext parent){
        ApplicationContext javaSEAppContext = AppContextHolder.getAppContext();
        GenericWebApplicationContext context = new GenericWebApplicationContext(servletContext);
        context.setParent(javaSEAppContext);
        return context;
    }
}

そしてもちろん、web.xmlを適応させます

    <!-- <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:application-context.xml</param-value>
    </context-param> -->
    <listener>
        <listener-class>com.company.MyContextLoaderListener</listener-class>
    </listener>
于 2013-01-30T16:47:17.087 に答える
0

もう 1 つの方法は、Bean を JNDI として登録してから、JNDI をルックアップして Bean のハンドルを取得することです。残念ながら、私が知る限り、Mbeans を登録する MBeanExporter はありますが、Bean をサーバーの JNDI ツリーにバインドするものはありません。

ただし、Spring には JNDI - JndiTemplateへのフックがあるため、顧客の JNDI エクスポーターの開発はそれほど難しくありません。そのため、DisposableBean を実装し、jndiName や jndiBean などの属性を持つクラスを作成し、以下を使用してバインドするだけです。 jndi 名に Bean を追加します。

JndiTemplate.bind()

春のxmlは次のようになります

<bean id="testBean" class="com.test.testBean"/>

<bean class="com.framework.JndiExporter">
   <property name="jndiBean" ref="testBean" />
   <property name="jndiName" value="com.test.testBean"/>
</bean>

これが登録されると、通常の方法を使用して、この jndiName の jndiTree を検索できます。

これがよりクリーンなアプローチを見つけるのに役立つことを願っています!

于 2013-01-30T18:22:19.070 に答える