7

我正在尝试从属性文件中读取属性,其文件名对于我们的每个环境都不同,例如 local.properties、dev.properties 等。这些属性文件将仅包含其相应 mongodb 实例的连接信息,例如主机、端口和数据库名称。通常这种事情会在我们的应用服务器中使用 JNDI 定义来完成,但目前还没有针对 Mongo 的实现。

由于我使用的是 WebLogic 10.3.6,因此我无法使用 Servlet 3.0 规范,因此无法使用 Spring 的 Java 配置,目前只能使用 XML。因此,我尝试使用的方法是在我的 web.xml 中定义一个 contextInitializerClass 上下文参数,然后将其设置为实现 ApplicationContextInitializer 并手动设置 Spring 活动配置文件的类。但是,在 WebLogic 启动或重新部署时,都没有调用我的自定义初始化程序类,我的配置文件也没有设置。

我的问题是,Spring 的 contextInitializerClass 是否依赖于 Servlet 3.0,还是我还缺少其他东西?

我定义的代码:

web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee    http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">

<context-param>
    <param-name>contextInitializerClass</param-name>
    <param-value>com.myapp.spring.SpringContextProfileInit</param-value>
</context-param>

<!-- Location of spring context config -->
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>/WEB-INF/spring/appServlet/servlet-context.xml</param-value>
</context-param>
<servlet>
    <servlet-name>appServlet</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
</servlet>      
...

SpringContextProfileInit.java

public class SpringContextProfileInit implements ApplicationContextInitializer<ConfigurableWebApplicationContext> {

    private static final Logger log = LoggerFactory.getLogger(SpringContextProfileInit.class);

    public SpringContextProfileInit() {
        log.debug("Got the constructor");
    }

    @Override
    public void initialize(ConfigurableWebApplicationContext ctx) {
        ConfigurableWebEnvironment environ = ctx.getEnvironment();
        log.debug("Got the environment, no profiles should be set: "+ environ.getActiveProfiles());

        /*
        * Here I am setting the profile with a hardcoded name.  In the real app,
        * I would read from a separate properties file, always named app.properties
        * which would live on the app server's classpath.  That app.properties file
        * would contain a property directing the Spring Profile to use.
        */
        environ.setActiveProfiles("local");
        log.debug("Now should be set to local: "+ environ.getActiveProfiles());
        ctx.refresh();
    }

}

servlet-context.xml

<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context"
    xmlns:mongo="http://www.springframework.org/schema/data/mongo" 
    xmlns:c="http://www.springframework.org/schema/c"
xmlns:p="http://www.springframework.org/schema/p" 
xmlns:jpa="http://www.springframework.org/schema/data/jpa"
xmlns:cache="http://www.springframework.org/schema/cache"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="
    http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd
http://www.springframework.org/schema/data/mongo
http://www.springframework.org/schema/data/mongo/spring-mongo-1.0.xsd
    http://www.springframework.org/schema/data/jpa
    http://www.springframework.org/schema/data/jpa/spring-jpa.xsd
    http://www.springframework.org/schema/cache http://www.springframework.org/schema/cache/spring-cache.xsd
    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.1.xsd
   http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-3.1.xsd">
<context:property-placeholder properties-ref="deployProperties" />
...
<beans profile="local">
    <bean id="deployProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean"
            p:location="WEB-INF/local.properties" />
</beans>
<beans profile="beast, dev">
    <bean id="deployProperties" class="org.springframework.beans.factory.config.PropertiesFactoryBean"
            p:location="WEB-INF/dev.properties" />
</beans>
</beans>

当我尝试部署应用程序时,出现异常: NoSuchBeanDefinitionException:No bean named 'deployProperties' is defined 如果未设置配置文件,这是预期的。我的日志没有显示我的任何调试语句都已打印。我还尝试将 contextInitializerClass 参数移动为我的 DispatcherServlet 的 init-param,但这给出了相同的结果。

我的限制是

  1. 我无法从我的 Maven 脚本中设置配置文件,因为我们公司使用相同的工件来推送到所有环境。

  2. 我也无法更改 WebLogic 的版本或使用最新的 servlet 规范,因为它依赖于容器。

我目前的版本是:

  • Spring 3.1.2.RELEASE
  • WebLogic 10.3.6
  • javax.servlet-api 2.5

有没有其他人看到这个问题并且知道如何加载我的初始化程序类?还是有更好的方法来做我想做的事情?

这看起来与另一个尚未回答的发帖者的问题有关:Spring MVC 3.1 Using Profiles for environment specific Hibernate settings

4

1 回答 1

7

我认为 context-param 的名称是错误的,应该contextInitializerClasses不是contextInitializerClass,这可能是您的 ApplicationContextInitializer 没有被拾取的原因

此外,您似乎缺少ContextLoaderListenerweb.xml 文件中的条目,也请尝试添加:

<listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

这是加载contextConfigLocation标签中指定的bean配置xml文件的文件

于 2012-11-04T00:57:23.167 回答