メモリ使用量とCPU使用率を追跡するスレッドを作成したいと思います。
アプリケーションが高レベルに達した場合、ヒープダンプまたはスレッドダンプを生成したいと思います。
再起動せずにスレッドダンプランタイムを生成する方法はありますか?
メモリ使用量とCPU使用率を追跡するスレッドを作成したいと思います。
アプリケーションが高レベルに達した場合、ヒープダンプまたはスレッドダンプを生成したいと思います。
再起動せずにスレッドダンプランタイムを生成する方法はありますか?
プログラムでそれを行う方法は次のとおりです:http://pastebin.com/uS5jYpd4
JMX
ThreadMXBean
およびThreadInfo
クラスを使用します。
ThreadMXBean mxBean = ManagementFactory.getThreadMXBean();
ThreadInfo[] threadInfos = mxBean.getThreadInfo(mxBean.getAllThreadIds(), 0);
...
kill -QUIT pid
under〜unixを実行して、スタックを標準出力にダンプすることもできます。JVMのスタックをダンプするjstackもあります。
また、アプリケーションの負荷平均があるしきい値を超えた場合にスタックをダンプする自動化もあります。
private long lastCpuTimeMillis;
private long lastPollTimeMillis;
public void checkLoadAverage() {
long now = System.currentTimeMillis();
long currentCpuMillis = getTotalCpuTimeMillis();
double loadAvg = calcLoadAveragePercentage(now, currentCpuMillis);
if (loadAvg > LOAD_AVERAGE_DUMP_THRESHOLD) {
try {
dumpStack("Load average percentage is " + loadAvg);
} catch (IOException e) {
// Oh well, we tried
}
}
lastCpuTimeMillis = currentCpuMillis;
lastPollTimeMillis = now;
}
private long getTotalCpuTimeMillis() {
long total = 0;
for (long id : threadMxBean.getAllThreadIds()) {
long cpuTime = threadMxBean.getThreadCpuTime(id);
if (cpuTime > 0) {
total += cpuTime;
}
}
// since is in nano-seconds
long currentCpuMillis = total / 1000000;
return currentCpuMillis;
}
private double calcLoadAveragePercentage(long now, long currentCpuMillis) {
long timeDiff = now - lastPollTimeMillis;
if (timeDiff == 0) {
timeDiff = 1;
}
long cpuDiff = currentCpuMillis - lastCpuTimeMillis;
double loadAvg = (double) cpuDiff / (double) timeDiff;
return loadAvg;
}
スレッドを標準にダンプするには、次のようにします。
ThreadInfo[] threads = ManagementFactory.getThreadMXBean()
.dumpAllThreads(true, true);
for (ThreadInfo info : threads) {
System.out.print(info);
}
Java6ではThreadMXBeanクラスを使用します。ただし、標準出力の代わりに実際のロギングを使用することをお勧めします。
「kill–QUIT」Process_idを試してください。例:
kill -QUIT 2134
これにより、再起動せずにスレッドダンプがトリガーされます
はい、組み込みの管理MXBeanを使用して独自のスタックダンプを生成できます。具体的には、 ThreadMXBeanから現在のすべてのThreadInfoを取得し、その内容を目的の場所に書き込むことができます。
Springアプリケーションでは、cronジョブを作成することにより、プログラムでヒープダンプを生成できます。
import com.sun.management.HotSpotDiagnosticMXBean;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import javax.management.MBeanServer;
import java.io.File;
import java.lang.management.ManagementFactory;
@Component
public class HeapDumpGenerator {
@Scheduled(cron = "0 0/5 * * * *")
public void execute() {
generateHeapDump();
}
private void generateHeapDump() {
MBeanServer mBeanServer = ManagementFactory.getPlatformMBeanServer();
String dumpFilePath = System.getProperty("user.home") + File.separator +
"heap-dumps" + File.separator +
"dumpfile_" + System.currentTimeMillis() + ".hprof";
try {
HotSpotDiagnosticMXBean hotSpotDiagnosticMXBean = ManagementFactory
.newPlatformMXBeanProxy(mBeanServer,
"com.sun.management:type=HotSpotDiagnostic",
HotSpotDiagnosticMXBean.class);
hotSpotDiagnosticMXBean.dumpHeap(dumpFilePath, Boolean.TRUE);
} catch (Exception e) {
// log error
}
}
}
5分ごとに実行されますが、独自の時間/間隔を設定できます。メソッドを呼び出す前に条件を追加することもできますgenerateHeapDump()
。