java - Log4j 挂起我的应用程序我做错了什么?

标签 java multithreading log4j deadlock

首先是应用程序的一些背景。我有一个应用程序通过线程池并行处理许多独立任务。线程池现在挂了。

以下是我的线程转储中的一个片段,我在 pool-2 中的所有线程都被“pool-2-thread-78”阻塞。它似乎被锁定试图写入我觉得非常奇怪的控制台。任何人都可以为我阐明情况吗?

编辑: 平台详情 Java 版本“1.6.0_07” Java(TM) SE 运行时环境(build 1.6.0_07-b06) Java HotSpot(TM) 客户端 VM(build 10.0-b23,混合模式,共享)

Ubuntu Linux 服务器双四核机器。

它似乎在写入打印流时锁定,我考虑过只删除控制台附加程序,但我更想知道它为什么阻塞并根据此知识将其删除。在过去,删除并查看它是否有效已经回来咬我:)

我的 log4j 中的相关部分

log4j.rootLogger=DEBUG, STDOUT log4j.logger.com.blah=INFO, LOG log4j.appender.STDOUT=org.apache.log4j.ConsoleAppender log4j.appender.LOG=org.apache.log4j.FileAppender

线程转储提取

"pool-2-thread-79" Id=149 BLOCKED on org.apache.log4j.spi.RootLogger@6c3ba437 owned by "pool-2-thread-78" Id=148 at org.apache.log4j.Category.callAppenders(Category.java:201) at org.apache.log4j.Category.forcedLog(Category.java:388) at org.apache.log4j.Category.error(Category.java:302) at com.blah.MessageProcessTask.run(MessageProcessTask.java:103) at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:441) at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask/java:268) at java.util.concurrent.FutureTask.run(FutureTask/java:54) at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:885) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:907) at java.lang.Thread.run(Thread.java:619)

"pool-2-thread-78" Id=148 RUNNABLE at java.io.FileOutputStream.writeBytes(Native Method) at java.io.FileOutputStream.write(FileOutputStream.java:260) at java.io.BufferedOutputStream.write(BufferedOutputStream.java:105) - locked <0x6f314ba4> (a java.io.BufferedOutputStream) at java.io.PrintStream.write(PrintStream.java:430) - locked <0xd5d3504> (a java.io.PrintStream) at org.apache.log4j.ConsoleAppender$SystemOutStream.write(ConsoleAppender.java:173) at sun.nio.cs.StreamEncoder.writeBytes(StreamEncoder.java:202) at sun.nio.cs.StreamEncoder.implFlushBuffer(StreamEncoder.java:272) at sun.nio.cs.StreamEncoder.implFlush(StreamEncoder.java:276) at sun.nio.cs.StreamEncoder.flush(StreamEncoder.java:122) - locked <0x6243a076> (a java.io.OutputStreamWriter) at java.io.OutputStreamWriter.flush(OutputStreamWriter.java:212) at org.apache.log4j.helpers.QuietWriter.flush(QuietWriter.java:57) at org.apache.log4j.WriterAppender.subAppend(WriterAppender.java:315) at org.apache.log4j.WriterAppender.append(WriterAppender.java:159) at org.apache.log4j.AppenderSkeleton.doAppend(AppenderSkeleton.java:230) - locked <0x45dbd560> (a org.apache.log4j.ConsoleAppender) at org.apache.log4j.helpers.AppenderAttachableImpl.appendLoopOnAppenders(AppenderAttachableImpl.java:65) at org.apache.log4j.Category.callAppenders(Category.java:203) - locked <0x6c3ba437> (a org.apache.log4j.spi.RootLogger) at org.apache.log4j.Category.forcedLog(Category.java:388) at org.apache.log4j.Category.error(Category.java:302) at com.blah.MessageProcessTask.run(MessageProcessTask.java:103) at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:441) at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask/java:268) at java.util.concurrent.FutureTask.run(FutureTask/java:54) at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:885) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:907) at java.lang.Thread.run(Thread.java:619)

最佳答案

您可以使用 AsyncAppender 更好地将记录器与附加程序分离。

在 Windows 上,如果您在控制台窗口中单击,这将暂停控制台,例如stdout 缓冲区将填满,并且随着控制台 appender 串行写入,您的应用程序将挂起,直到您释放控制台(按 enter 左右)。

考虑将 AsyncAppender 与 log4j 一起使用 - 大多数时候这是个好主意 - 唯一的问题是 - AsynAppender 缓冲区在退出时没有完全刷新。

关于java - Log4j 挂起我的应用程序我做错了什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/634102/

相关文章:

基于每个连接的 Java Authenticator?

java - Java 中的同步 - Thinking in Java 示例

java线程同步问题

spring - 登录Grails:使用Log4j2重新加载外部配置

java - Java中log4j基于级别的日志记录

java - 比较两点

java - net.sf.json序列化为json时如何添加类名

java - Log4j2 中的 Socket Appender 出现错误

Java:在同一线程上多次调用 join()?

c++ - Qt中正确的线程方式