遇到了同样的问题,

我的解决办法是在pom.xml中增加如下配置,去除对于jcl-over-slf4j.jar的依赖。

    <exclusions>
                <exclusion>
                    <artifactId>jcl-over-slf4j</artifactId>
                    <groupId>org.slf4j</groupId>
                </exclusion>
            </exclusions>

顺便转载下大神解答:

转载自:http://blog.csdn.net/xian00000/article/details/10013395

今天启动tomcat服务失败,碰到异常情况如下

java.lang.NoSuchMethodError: org.slf4j.spi.LocationAwareLogger.log(Lorg/slf4j/Marker;Ljava/lang/String;ILjava/lang/String;Ljava/lang/Throwable;)V
    at org.apache.commons.logging.impl.SLF4JLocationAwareLog.info(SLF4JLocationAwareLog.java:159)

Commons-log + log4j 这黄金搭档一直以来都让我们很省心,很好的完成了日志的需求。但是随着技术的变更和对性能的追求,slf4j 和 logback 这对后起之秀的到来好像打破了原本很平静的日志系统,频繁的出现包冲突...

和平的日子不在了,让我们一起来看看究竟发生了什么...

首先看看这些个包,特别是slf4j引入后就引入了一大堆包之后就有点懵了。

为什么commons-logging和jcl-over-slf4j会有冲突呢?看一下它们的类结构

很清晰的可以看到jcl-over-slf4j 重写了 commons-logging...

还有slf4j-api的实现呢,同样看类:

其实就这么简单,往往看了代码之后才发现错误是这么显而易见。。。

顺着研究,继续看一下slf4j的源码及流程

1.测试类

java代码

  1. package com.taobao.wuzhong.log;
  2. import org.apache.commons.logging.Log;
  3. import org.apache.commons.logging.LogFactory;
  4. import org.junit.Test;
  5. import org.slf4j.Logger;
  6. import org.slf4j.LoggerFactory;
  7. /**
  8. * DESC:
  9. *
  10. * Copyright: Copyright 2011 m.taobao.com
  11. *
  12. * @author wuzhong@taobao.com
  13. * @time 2011-4-6 下午03:42:11
  14. * @version 1.0
  15. **/
  16. public class LogTest {
  17. // Logback tries to find a file called logback.groovy in the classpath.
  18. // If no such file is found, logback tries to find a file called
  19. // logback-test.xml in the classpath.
  20. // If no such file is found, it checks for the file logback.xml in the
  21. // classpath..
  22. // If neither file is found, logback configures itself automatically using
  23. // the BasicConfigurator which will cause logging output to be directed to
  24. // the console.
  25. @Test
  26. public void test() {
  27. //commons-logging的方式获取
  28. Log log = LogFactory.getLog(LogTest.class);
  29. //slf4j直接的方式获取,推荐用这个
  30. Logger log2 = LoggerFactory.getLogger(LogTest.class);
  31. log.debug("eeeeee {} {} {}");
  32. log2.debug("{} {} {}", new String[] { "a", "b", "c" });
  33. }
  34. }

logFactory.getLog 会调用内部静态变量 Slf4jLogFactory.getInstance方法,如下:

public Log getInstance(String name) throws LogConfigurationException {

  1. Log instance = null;
  2. // protect against concurrent access of loggerMap
  3. synchronized (this) {
  4. instance = (Log) loggerMap.get(name);
  5. if (instance == null) {
  6. Logger logger = LoggerFactory.getLogger(name);   //slf4j的方式,代理过去了
  7. if(logger instanceof LocationAwareLogger) {
  8. instance = new SLF4JLocationAwareLog((LocationAwareLogger) logger);  //包装了一层,做适配
  9. } else {
  10. instance = new SLF4JLog(logger);
  11. }
  12. loggerMap.put(name, instance);
  13. }
  14. }
  15. return (instance);

loggerFactory 会调用getILoggerFactory().getlOgger()

  1. LoggerFactory.java
  2. public static ILoggerFactory getILoggerFactory() {
  3. if (INITIALIZATION_STATE == UNINITIALIZED) {
  4. INITIALIZATION_STATE = ONGOING_INITILIZATION;
  5. performInitialization();
  6. }
  7. switch (INITIALIZATION_STATE) {
  8. case SUCCESSFUL_INITILIZATION:
  9. return getSingleton().getLoggerFactory();
  10. case FAILED_INITILIZATION:
  11. throw new IllegalStateException(UNSUCCESSFUL_INIT_MSG);
  12. case ONGOING_INITILIZATION:
  13. // support re-entrant behavior.
  14. // See also http://bugzilla.slf4j.org/show_bug.cgi?id=106
  15. return TEMP_FACTORY;
  16. }
  17. throw new IllegalStateException("Unreachable code");
  18. }
  19. private final static void performInitialization() {
  20. bind();
  21. versionSanityCheck();
  22. singleImplementationSanityCheck();
  23. }

这里的bind很关键,这里动态的绑定了slf4j-api的实现机制

  1. static {
  2. SINGLETON.init();
  3. }
  4. /**
  5. * Package access for testing purposes.
  6. */
  7. void init() {
  8. try {
  9. try {
  10. new ContextInitializer(defaultLoggerContext).autoConfig();
  11. } catch (JoranException je) {
  12. Util.reportFailure("Failed to auto configure default logger context",
  13. je);
  14. }
  15. StatusPrinter.printInCaseOfErrorsOrWarnings(defaultLoggerContext);
  16. contextSelectorBinder.init(defaultLoggerContext, KEY);
  17. initialized = true;
  18. } catch (Throwable t) {
  19. // we should never get here
  20. Util.reportFailure("Failed to instantiate ["
  21. + LoggerContext.class.getName() + "]", t);
  22. }
  23. }

获取配置信息初始化

  1. autoConfig ….
  2. public URL findURLOfDefaultConfigurationFile(boolean updateStatus) {
  3. ClassLoader myClassLoader = Loader.getClassLoaderOfObject(this);
  4. URL url = findConfigFileURLFromSystemProperties(myClassLoader, updateStatus);
  5. if (url != null) {
  6. return url;
  7. }
  8. url = Loader.getResource(TEST_AUTOCONFIG_FILE, myClassLoader);
  9. if (updateStatus) {
  10. statusOnResourceSearch(TEST_AUTOCONFIG_FILE, myClassLoader, url);
  11. }
  12. if (url != null) {
  13. return url;
  14. }
  15. url = Loader.getResource(AUTOCONFIG_FILE, myClassLoader);
  16. if (updateStatus) {
  17. statusOnResourceSearch(AUTOCONFIG_FILE, myClassLoader, url);
  18. }
  19. return url;
  20. }
  21. public void autoConfig() throws JoranException {
  22. StatusListenerConfigHelper.installIfAsked(loggerContext);
  23. URL url = findURLOfDefaultConfigurationFile(true);
  24. if (url != null) {
  25. configureByResource(url);
  26. } else {
  27. BasicConfigurator.configure(loggerContext);
  28. }
  29. }

最后画张流程图总结下,^_^

总结: log框架应该很好的诠释了 facade , adapter , 实现上还是比较简单的,很好的做到了接口和实现的分离,对今后的代码组织有一定的启发

转载自 : http://myclqr.iteye.com/blog/1775541

------------------------------------------------------------------------------------

用slf4j+logback替代commons-logging+log4j
加载以下jar包:
slf4j-api.jar
logback-core.jar
logback-classic.jar
log4j-over-slf4j.jar
jcl104-over-slf4j.jar
同时删除commons-logging.jar和log4j.jar
http://logback.qos.ch/translator/Welcome.do转换log4j.properties为logback.xml

log日志框架和LocationAwareLogger问题的更多相关文章

  1. Spring Boot Log 日志使用教程

    我们编写任何 Spring Boot 程序,可能绕不开的就是 log 日志框架(组件). 在大多数程序员眼中日志是用来定位问题的.这很重要. 本项目源码下载 注意本项目提供的源码已在后期重新编写,有部 ...

  2. scrapy框架之log日志

    scrapy中的debug信息 在scrapy中设置log 1.在settings中设置log级别,在settings.py中添加一行: Scrapy提供5层logging级别: CRITICAL - ...

  3. Xposed框架Hook Android应用的所有类方法打印Log日志

    本文博客地址:https://blog.csdn.net/QQ1084283172/article/details/80954759 在进行Android程序的逆向分析的时候,经常需要Android程 ...

  4. Atitit.log日志技术的最佳实践attilax总结

    Atitit.log日志技术的最佳实践attilax总结 1. 日志的意义与作用1 1.1. 日志系统是一种不可或缺的单元测试,跟踪调试工具1 2. 俩种实现[1]日志系统作为一种服务进程存在 [2] ...

  5. 解读ASP.NET 5 & MVC6系列(9):日志框架

    框架介绍 在之前的.NET中,微软还没有提供过像样的日志框架,目前能用的一些框架比如Log4Net.NLog.CommonLogging使用起来多多少少都有些费劲,和java的SLF4J根本无法相比. ...

  6. Java日志框架:SLF4J,Common-Logging,Log4J,Logback说明

    Log4j  Apache的一个开放源代码项目,通过使用Log4j,我们可以控制日志信息输送的目的地是控制台.文件.GUI组件.甚至是套接口服务 器.NT的事件记录器.UNIX Syslog守护进程等 ...

  7. java日志框架slf4j与log4j

    日志记录自然是非常重要的,但恐怕能记住slf4j与log4j等日志框架配置的人就很少了,这个东西不难,只是配置好后很少会去动它,开发新项目一般也是从其他项目拷贝,或者参照文档 废话不多说,先说log4 ...

  8. Moon转告给你一个比Log4net更好日志框架--TracerX Logger 及其对应的日志查看器

    一.介绍 TracerX logger是一个易于上手,且拥有众多高级特性的.NET日志框架. 它能够发送输出结果到多目的地(循环文件.事件日志等....).它也能生成文本和二进制文件.它拥有一个强大的 ...

  9. lombok+slf4j+logback SLF4J和Logback日志框架详解

    maven 包依赖 <dependency> <groupId>org.projectlombok</groupId> <artifactId>lomb ...

随机推荐

  1. UICollectionViews有了简单的重排功能

    代码地址如下:http://www.demodashi.com/demo/13213.html 一.前言 我是UICollectionView的忠实粉丝.这个类比起它的老哥UITableView类具有 ...

  2. openstack架构简单介绍J版(更新中)

    title : OPENSTACK架构简单介绍 openstack的发展及历史 openstack是什么? OpenStack是一个美国国家航空航天局和Rackspace合作研发的云端运算‎软件,以A ...

  3. JS的同步加载、异步加载

    在使用js展开式菜单时,发现只有加载完页面包含的js文件时,展开菜单才能折叠起来. 查找了一下原因:是因为js页面加载使用的是同步模式,又称阻塞模式,会阻止浏览器的后续处理,停止后续的解析,只有当当前 ...

  4. Eclipse RCP应用开发(概念)

    学习Eclipse RCP应用开发,对其中一些相关的概念需要了解. 让我们在讲解概念前先思考两个问题. 问题一.什么是Eclipse 大部分同学会回答eclipse是IBM公司开发的java集成开发环 ...

  5. Python内置函数之chr()

    参数是一个整型对象: chr(integer) 返回整型对应的Unicode字符对象. 例子: >>> chr() '\x0c' >>> chr() 'ⲓ' > ...

  6. python 属性 property、getattr()、setattr()详解

    直奔主题 使用中文注释需要使用 #-*-coding:utf-8-*- property property在python中有2中使用property方法:1.@property @属性名称.sette ...

  7. DataUml Design 介绍9 - DataUML 1.3版本功能(查询分析器功能等)

    DataUML 1.3 (下载)主要更新内容如下: 1.增加查询分析器功能: 2.增加打开历史文件记录功能: 3.修改查询对象功能: 4.增加显示对象长度功能: 5.增加配置显示表字段功能: 6.增加 ...

  8. openWRT自学---针对backfire版本的主要目录和文件的作用的分析整理

    特别说明:要编译backfire版本,一定要通过svn下载:svn co svn://svn.openwrt.org/openwrt/branches/backfire,而不能使用http://dow ...

  9. Linux下printf函数显示不同的颜色(转)

    Linux下printf函数显示不同的颜色 在学习Linux网络编程的时候做一个聊天系统,当时为了界面更漂亮点,于是搜索了下关于printf()函数的用法,给printf的输出加上些特效比如颜色,可以 ...

  10. 应用市场出新规,国内千款应用TargetSdkVersion调查分析

    ​​​近期,谷歌针对Google Play上架应用发布强制性适配Android 8.0公告,要求自2018 年 8 月 1 日起,所有向 Google Play 首次提交的新应用都必须基于 Andro ...