Giter Club home page Giter Club logo

logback-mdc-ttl's Introduction

logback-mdc-ttl

Maven Central GitHub release License

👉 logback扩展,支持跨线程池的mdc跟踪。

实现上集成使用了Transmittable ThreadLocal(TTL) :在使用线程池等会缓存线程的组件情况下,提供ThreadLocal值的传递功能,解决异步执行时上下文传递的问题。支持JDK 9/8/7/6。

示例

1.项目依赖

<dependency>
    <groupId>com.ofpay</groupId>
    <artifactId>logback-mdc-ttl</artifactId>
    <version>1.0.2</version>
</dependency>

2. 在Java的启动参数加上:

  • -Xbootclasspath/a:/path/to/transmittable-thread-local-2.x.x.jar
  • -javaagent:/path/to/transmittable-thread-local-2.x.x.jar

3. 在logback配置文件中增加TtlMdcListener

<?xml version="1.0" encoding="UTF-8"?>
<configuration >
    <!-- ...(略) -->
    <contextListener class="com.ofpay.logback.TtlMdcListener"/>

    <!--例子:  %X{uuid} 支持在跨线程池时传递-->
    <property scope="context" name="APP_PATTERN"
              value='%d{yyyy-MM-dd HH:mm:ss.SSS}|%X{uuid}|%level|%M|%C\:%L|%thread|%replace(%.-2000msg){"(\r|\n)","\t"}|"%.-2000ex{full}"%n'/>
</configuration>

logback-mdc-ttl's People

Contributors

oldratlee avatar wuwen5 avatar

Stargazers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

Watchers

 avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar  avatar

logback-mdc-ttl's Issues

slf4j+logback的MDC原生就支持多线程的困惑

我在验证slf4j+logback的MDC时,是支持在多线程中输出;还需要你这种扩展么?

 @Test
    public void run32() throws IOException {
        String token = UUID.randomUUID().toString();

        MDC.put("G_TOKEN", token);
        log.debug("token is {}", token);

        new Thread(new Runnable() {
            @Override
            public void run() {
                log.debug("thread-2 start");

                log.debug("map={}", MDC.getCopyOfContextMap());

                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        log.debug("thread-3 start");

                        log.debug("map={}", MDC.getCopyOfContextMap());


                        Runnable runable = new Runnable() {
                            @Override
                            public void run() {
                                log.debug("abc");
                            }
                        };

                        ExecutorService executor = Executors.newFixedThreadPool(1);
                        executor.submit(runable);
                    }
                }).start();

            }
        }).start();


        System.in.read();
        MDC.clear();
    }

输出

2017/07/20 13:23:15.433 DEBUG [main][7f7f35f0-7adf-4792-810d-0a6c56502b04] com.spy.test.log.MDCTest:39>>token is 7f7f35f0-7adf-4792-810d-0a6c56502b04
2017/07/20 13:23:15.437 DEBUG [Thread-0][7f7f35f0-7adf-4792-810d-0a6c56502b04] com.spy.test.log.MDCTest$1:44>>thread-2 start
2017/07/20 13:23:15.437 DEBUG [Thread-0][7f7f35f0-7adf-4792-810d-0a6c56502b04] com.spy.test.log.MDCTest$1:46>>map={G_TOKEN=7f7f35f0-7adf-4792-810d-0a6c56502b04}
2017/07/20 13:23:15.439 DEBUG [Thread-1][7f7f35f0-7adf-4792-810d-0a6c56502b04] com.spy.test.log.MDCTest$1$1:51>>thread-3 start
2017/07/20 13:23:15.440 DEBUG [Thread-1][7f7f35f0-7adf-4792-810d-0a6c56502b04] com.spy.test.log.MDCTest$1$1:53>>map={G_TOKEN=7f7f35f0-7adf-4792-810d-0a6c56502b04}
2017/07/20 13:23:15.446 DEBUG [pool-2-thread-1][7f7f35f0-7adf-4792-810d-0a6c56502b04] com.spy.test.log.MDCTest$1$1$1:59>>abc

springboot tomcat 线程池 发现一个mdc串的问题

transmittable-thread-local 2.6.0
logback-mdc-ttl 1.0.2
logback-classic 1.2.3

-javaagent:/Users/transmittable-thread-local-2.6.0.jar -Xbootclasspath/a:/Users/transmittable-thread-local-2.6.0.jar

springboot 2.0 运行一段时间之后. tomcat 线程池里面的线程 都已经 初始化完毕.
这个时候

两个HTTP请求
A请求, 第0秒开始执行, 第20s执行完毕.
B请求, 第10秒开始执行, 第12秒执行完毕.

第5秒, A请求打的 mdc里面的 traceId 是 111
第12秒, B请求打的 mdc里面的 traceId 是 222
第18秒,A请求打的mdc里面的 traceId 是 222 (问题出现在这里.)

辛苦看一下

slfj-1.4.1版本不支持

该版本新增了很多方法,导致没有实现方法,启动报错
image
image

备注:我使用的是Spring Boot 3.2版本,升级后logback、slfj版本也都升级了,导致失效。

翻看logback-classic源码,发现1.2.11版本下是这样拿Map的:
image
但是在高版本是这样的,所以失效了:
image

logback1.2.3似乎不支持MDC线程可继承了

logback从1.1.5版本之后,新的commit将MDC内部维护的InheritThreadLocal修改成了ThreadLocal,MDC不支持线程继承了。

Child threads no longer inherit MDC values. In previous versions of logback as well as log4j 1.x, MDC values were inherited by child threads. Several users have argued convincingly that MDC inheritance by child threads was unhelpful and even dangerous. This change fixes LOGBACK-422 and LOGBACK-624
https://logback.qos.ch/news.html

MDC在多线程下无法正确传值

使用了

transmittable-thread-local 2.8.1 & logback-mdc-ttl 1.0.2

代码示例:

TtlMDCAdapter.getInstance();
int nThreads = 10;
ExecutorService executorService = Executors.newFixedThreadPool(nThreads);
for (int i = 0; i < nThreads; i++) {
	MDC.put("key", "value:" + i);
	executorService.execute(() -> {
		int random = new Random().nextInt(1000);
		try {
			TimeUnit.MILLISECONDS.sleep(random);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		System.out.println(MDC.get("key"));
	});
}
TimeUnit.SECONDS.sleep(3);

期望输出:

value:0
value:2
value:1
xxx

value 0-9无重复数字

实际输出:

value:9
value:9
value:9
value:9
value:9
value:9
value:9
value:9
value:9

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. 📊📈🎉

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google ❤️ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.