spring - 如何在不使用命令行上的 -javaagent 的情况下使用 Spring 和 Tomcat 设置加载时编织

标签 spring tomcat spring-aop context.xml load-time-weaving

我正在使用 Spring 3.2.9,Tomcat 6.0.44

当部署在 Tomcat 上时,我正在尝试配置我的应用程序的 Spring 检测提供程序(例如 spring-instrumentation.jar)以进行加载时织入。

我有一个不使用的要求: "-javaagent:/path/path/spring-instrument.jar"在命令行上进行配置。

我读到我可以通过修改应用程序 Tomcat 配置的 元素来配置 Spring 检测(在 Tomcat 的 server.xml 或我的 Web 应用程序的 context.xml 中)。将适当的 元素添加到 server.xml 会导致我的应用程序被正确配置为与 Spring 的检测提供程序一起运行。将它添加到 context.xml(见下文)不会导致工作设置。

我有一个 META-INF/aop.xml 文件,看起来像这样:

    <aspectj>
        <weaver options="-verbose -showWeaveInfo -debug">
            <include within="com.mv.xx.services..*"/>
            <exclude within="com.mv.xx.aop..*"/>
        </weaver>
        <aspects>
            <aspect name="com.mv.xx.aop.MyAspect"/>
        </aspects>
    </aspectj>

我还通过将此添加到我的 Spring 上下文配置来指定我想使用加载时编织:

 <context:load-time-weaver />

然后我将这个 jar 添加到我的应用程序的类路径中: spring-instrument-tomcat.jar

我尝试过的:

启动 Tomcat 时,如果我在命令行上使用 -javaagent 参数确定 spring-instrument.jar 的位置,如下所示:

-javaagent:/path/path2/spring-instrument-3.2.9.RELEASE.jar

一切正常。


接下来,我从命令行中删除了“-javaagent:/path/path2/spring-instrument-3.2.9.RELEASE.jar”。 在 Tomcat 的 server.xml 文件(位于 $CATALINE_HOME/conf)中,我向 元素添加了一个 元素,如下所示:

<Context path="/myApp" docBase="myApp">
    <Loader loaderClass="org.springframework.instrument.classloading.tomcat.TomcatInstrumentableClassLoader"/>
</Context>

使用此配置,一切正常。但是我有一个不修改 Tomcat 的 server.xml 的要求,因为我无法控制 server.xml(DevOps 控制,并且不愿意修改它)。


接下来,我从 Tomcat 的 server.xml 中删除了 元素。 根据Spring docs ,我可以在我的 webapp 中添加一个/META-INF/context.xml ,并将曾经在 Tomcat 的 server.xml 中的 元素放入 context.xml 中,如下所示:

        <Context>
            <Context path="/myApp" docBase="myApp">
                <Loader loaderClass="org.springframework.instrument.classloading.tomcat.TomcatInstrumentableClassLoader"/>
            </Context>
        </Context>

但是,当我重新启动 Tomcat 时,我在日志中收到一条错误消息:

    SEVERE: Exception sending context initialized event to listener instance of class org.springframework.web.context.ContextLoaderListener
    org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'org.springframework.context.weaving.AspectJWeavingEnabler#0': Initialization of bean failed; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'loadTimeWeaver': Initialization of bean failed; nested exception is java.lang.IllegalStateException: ClassLoader [org.apache.catalina.loader.WebappClassLoader] does NOT provide an 'addTransformer(ClassFileTransformer)' method. Specify a custom LoadTimeWeaver or start your Java virtual machine with Spring's agent: -javaagent:org.springframework.instrument.jar

在四处挖掘之后,我读到一些建议我修改 <context:load-time-weaver/> 的内容我的 Spring 配置中的元素,如下所示:

        <context:load-time-weaver weaver-class="org.springframework.instrument.classloading.InstrumentationLoadTimeWeaver" />

并将包含 InstrumentationLoadTimeWeaver.class 的 jar 添加到我的类路径中。

但是,当我这样做时,我在日志中收到此错误消息:

        SEVERE: Exception sending context initialized event to listener instance of class org.springframework.web.context.ContextLoaderListener
        java.lang.IllegalStateException: Must start with Java agent to use InstrumentationLoadTimeWeaver. See Spring documentation.
    etc....

谁能解释如何在不在命令行上使用 -javaagent 且不修改 server.xml 的情况下使用 Spring 和 Tomcat 设置加载时编织?

最佳答案

这是我设法用来消除您提到的异常的代码。 基本上,您必须实现 LoadTimeWeavingConfigurer 并覆盖方法 getLoadTimeWeaver()。

@Configuration
@ComponentScan(basePackages = "org.myproject")
@EnableAspectJAutoProxy
@EnableSpringConfigured
@EnableLoadTimeWeaving(aspectjWeaving = EnableLoadTimeWeaving.AspectJWeaving.AUTODETECT)
public class Config implements LoadTimeWeavingConfigurer {

    @Override
    public LoadTimeWeaver getLoadTimeWeaver() {
        return new ReflectiveLoadTimeWeaver();
    }

    @Bean
    public InstrumentationLoadTimeWeaver loadTimeWeaver()  throws Throwable {
        InstrumentationLoadTimeWeaver loadTimeWeaver = new InstrumentationLoadTimeWeaver();
        return loadTimeWeaver;
    }

}

关于spring - 如何在不使用命令行上的 -javaagent 的情况下使用 Spring 和 Tomcat 设置加载时编织,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35993888/

相关文章:

java - maven:无法覆盖 tomcat 服务器的 server.port

java - 如何编写切面来满足不同的对象和方法参数

java - Spring注解在不同层的应用

java - Spring @Transactional 提交失败; Deby + Eclipse链接

Spring 3 构造函数 Autowiring - 为什么这段代码有效?

java - Tomcat7 如何使用 Expires HTTP header 提供静态文件

java - jackson- 对象列表 com.fasterxml.jackson.core.io.JsonEOFException : Unexpected end-of-input in field name?

java - 从 tomcat 取消部署 Web 应用程序不会删除已部署的目录

java - 非常简单的 Spring AOP 切入点

java - 为什么我的Web应用程序NoClassDefFoundError : org/aspectj/lang/reflect/AjTypeSystem