10

メモリ使用量とCPU使用率を追跡するスレッドを作成したいと思います。

アプリケーションが高レベルに達した場合、ヒープダンプまたはスレッドダンプを生成したいと思います。

再起動せずにスレッドダンプランタイムを生成する方法はありますか?

4

5 に答える 5

12

プログラムでそれを行う方法は次のとおりです:http://pastebin.com/uS5jYpd4

JMX ThreadMXBeanおよびThreadInfoクラスを使用します。

ThreadMXBean mxBean = ManagementFactory.getThreadMXBean();
ThreadInfo[] threadInfos = mxBean.getThreadInfo(mxBean.getAllThreadIds(), 0);
...

kill -QUIT pidunder〜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;
}
于 2012-10-11T14:55:17.760 に答える
10

スレッドを標準にダンプするには、次のようにします。

ThreadInfo[] threads = ManagementFactory.getThreadMXBean()
        .dumpAllThreads(true, true);
for (ThreadInfo info : threads) {
    System.out.print(info);
}

Java6ではThreadMXBeanクラスを使用します。ただし、標準出力の代わりに実際のロギングを使用することをお勧めします。

于 2012-10-11T14:55:02.940 に答える
2

「kill–QUIT」Process_idを試してください。例:

kill -QUIT 2134

これにより、再起動せずにスレッドダンプがトリガーされます

于 2013-05-19T06:05:56.350 に答える
0

はい、組み込みの管理MXBeanを使用して独自のスタックダンプを生成できます。具体的には、 ThreadMXBeanから現在のすべてのThreadInfoを取得し、その内容を目的の場所に書き込むことができます。

于 2012-10-11T14:54:27.330 に答える
0

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()

于 2019-08-04T11:03:20.813 に答える