使用EJB3.1进行logback

kgr*_*rad 6 java glassfish logback slf4j ejb-3.0

我正在使用logback/slf4j来处理我的应用程序中的日志记录.在我开始使用EJB之前,一切都运行良好.一旦我将无状态EJB添加到我的应用程序,记录器就开始忽略我的logback.xml并停止使用我的appender.我切换到编程记录器配置以查看错误,现在当我尝试在EJB中使用我的记录器时,我收到以下错误:

org.slf4j.impl.JDK14LoggerFactory cannot be cast to ch.qos.logback.classic.LoggerContext

来自这条线:

LoggerContext lc = (LoggerContext) LoggerFactory.getILoggerFactory();

是否需要任何特殊配置才能使回归与EJB一起使用?如果重要的话我将部署在glassfish v3上.

Pas*_*ent 9

这看起来非常接近此线程中描述的问题,我怀疑类似的类加载问题.由于logback加载的方式logback.xml(更准确地说是它检索的方式ClassLoader),它可能无法获取其配置文件并回退到默认值BasicConfiguration.

不确定如何打包代码,但建议的解决方法是将其包含logback.xml在EAR库中.如果您没有使用EAR打包,请尝试识别用于查看放置logback.xml文件位置的类加载器.

最后,这可能是回归中的问题.但是没有检查他们的问题跟踪器.

更新:如果您使用war包装,请尝试将GlassFish配置为在委托之前首先使用子类加载器.在sun-web.xml:

<sun-web-app>
  <class-loader delegate="false"/>
</sun-web-app>
Run Code Online (Sandbox Code Playgroud)


更新:我在我身边做了一点测试......我无法重现你的问题.我已经为Java EE 6 webapp创建了一个项目,它具有以下结构:

$ tree sample
sample
|-- pom.xml
`-- src
    `-- main
        |-- java
        |   `-- com
        |       `-- stackoverflow
        |           `-- q2418355
        |               |-- SimpleEJB.java
        |               `-- SimpleServlet.java
        |-- resources
        |   `-- logback.xml
        `-- webapp
            |-- META-INF
            |   `-- MANIFEST.MF
            |-- WEB-INF
            |   `-- lib
            `-- index.jsp

我的pom.xml看起来像:

<?xml version="1.0" encoding="UTF-8"?>
<project xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"
  xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <modelVersion>4.0.0</modelVersion>
  <groupId>com.stackoverflow.q2418355</groupId>
  <artifactId>sample</artifactId>
  <version>1.0-SNAPSHOT</version>
  <packaging>war</packaging>
  <name>sample Maven Webapp</name>
  <url>http://maven.apache.org</url>
  <dependencies>
    <dependency>
      <groupId>javax</groupId>
      <artifactId>javaee-api</artifactId>
      <version>6.0</version>
      <scope>provided</scope>
    </dependency>
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.7</version>
      <scope>test</scope>
    </dependency>
    <dependency>
      <groupId>ch.qos.logback</groupId>
      <artifactId>logback-classic</artifactId>
      <version>0.9.18</version>
    </dependency>
    <dependency>
      <groupId>org.slf4j</groupId>
      <artifactId>slf4j-api</artifactId>
      <version>1.5.11</version>
    </dependency>
  </dependencies>
  <build>
    <plugins>
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-compiler-plugin</artifactId>
        <version>2.0.2</version>
        <configuration>
          <source>1.6</source>
          <target>1.6</target>
        </configuration>
      </plugin>
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-war-plugin</artifactId>
        <version>2.1-beta-1</version>
        <configuration>
          <failOnMissingWebXml>false</failOnMissingWebXml>
        </configuration>
      </plugin>
    </plugins>
    <finalName>sample</finalName>
  </build>
</project>
Run Code Online (Sandbox Code Playgroud)

代码SimpleEJB.java是:

package com.stackoverflow.q2418355;

import javax.ejb.Stateless;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@Stateless
public class SimpleEJB {
    private static Logger logger = LoggerFactory.getLogger(SimpleEJB.class);

    public String sayHello(String name) {
        logger.debug(">> sayHello()");
        logger.debug("<< sayHello()");
        return "Hello " + name + "!!!";
    }
}
Run Code Online (Sandbox Code Playgroud)

代码SimpleServlet.java是:

package com.stackoverflow.q2418355;

import java.io.IOException;
import java.io.PrintWriter;

import javax.ejb.EJB;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

@WebServlet(urlPatterns = { "/SimpleServlet" })
public class SimpleServlet extends HttpServlet {
    @EJB
    SimpleEJB bean;

    private static Logger logger = LoggerFactory.getLogger(SimpleServlet.class);

    @Override
    public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {
        logger.debug(">> doGet()");
        PrintWriter out = response.getWriter();
        out.println("<html><body>");
        out.println("<h2>Serving at: " + request.getContextPath() + "</h2>");
        out.println("<h2>Invoking EJB: " + bean.sayHello("Duke") + "</h2>");
        out.println("</body></html>");
        logger.debug("<< doGet()");
    }
}
Run Code Online (Sandbox Code Playgroud)

代码index.jsp是:

<html>
<body>
<h2>Hello World!</h2>
Invoke the Servlet by clicking <a href="SimpleServlet">here</a>.
</body>
</html>
Run Code Online (Sandbox Code Playgroud)

logback.xml看起来像:

<configuration>
  <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
    <layout class="ch.qos.logback.classic.PatternLayout">
      <Pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</Pattern>
    </layout>
  </appender>

  <appender name="FILE" class="ch.qos.logback.core.FileAppender">
    <File>/tmp/logs/testFile.log</File>
    <Append>true</Append>

    <layout class="ch.qos.logback.classic.PatternLayout">
      <Pattern>%-4relative [%thread] %-5level %logger{35} - %msg%n</Pattern>
    </layout>
  </appender>

  <logger name="com.stackoverflow.q2418355" level="TRACE"/>

  <root level="debug">
    <appender-ref ref="STDOUT" />
    <appender-ref ref="FILE" />
  </root>
</configuration>
Run Code Online (Sandbox Code Playgroud)

logback.xml得到了属性加载,并在调用servlet时得到以下跟踪(取自我的日志文件):

10913 [http-thread-pool-8080-(1)] DEBUG com.stackoverflow.q2418355.SimpleServlet - >> doGet()
10928 [http-thread-pool-8080-(1)] DEBUG com.stackoverflow.q2418355.SimpleEJB - >> sayHello()
10928 [http-thread-pool-8080-(1)] DEBUG com.stackoverflow.q2418355.SimpleEJB - << sayHello()
10932 [http-thread-pool-8080-(1)] DEBUG com.stackoverflow.q2418355.SimpleServlet - << doGet()
Run Code Online (Sandbox Code Playgroud)

我也尝试将EJB打包在自己的JAR中并部署WEB-INF/lib并获得相同的结果,它只是起作用.你能发现任何明显的区别吗?也许上传你的应用程序的简化版本(很可能是错误报告BTW所必需的).

我在Eclipse 3.5(使用GlassFish v3插件)下运行GlassFish v3.


Cek*_*eki 6

org.slf4j.impl.JDK14LoggerFactory cannot be cast to ch.qos.logback.classic.LoggerContext异常表明SLF4J未绑定到的logback经典但SLF4J-JDK14.简而言之,回归代码不应该受到责备,因为它没有被执行也没有被调用.

看起来GFv3正在将slf4j-jdk14.jar导出到您的应用程序中,从而覆盖了您选择的日志后端,在这种情况下是logback.这是app服务器无意中对用户施加选择的场景之一.

如果GFv3确实对最终用户施加了SLF4J绑定,那么这就是GFv3开发人员必须解决的GFv3问题.我可能错了,但我认为他们认为最终用户不希望任何其他日志记录功能超出java.util.logging提供的功能,只需在GFv3中捆绑slf4j-jdk14.用户需要联系他们并抱怨他们的假设不正确.他们也有可能意识到这个问题并且已经提供了解决方法......