ホームページ  >  記事  >  Java  >  Java の ThreadLocal がメモリ オーバーフロー (メモリ不足) を引き起こす理由は何ですか?

Java の ThreadLocal がメモリ オーバーフロー (メモリ不足) を引き起こす理由は何ですか?

WBOY
WBOY転載
2023-04-23 23:58:091052ブラウズ

原因分析

ThreadLocal がメモリ OOM を引き起こす理由は何ですか?

ThreadLocal の基礎となる層は、ThreadLocalMap を通じてデータを保存します。

ソース コードは次のとおりです。

Java の ThreadLocal がメモリ オーバーフロー (メモリ不足) を引き起こす理由は何ですか?

  • ThreadLocal.set() を使用すると、セット (つまり、ビジネス自体によって定義された ThreadLocal クラス) の値とキーが ThreadLocalMap の Entry[] 配列に格納されます

ソース コードは次のとおりです。

Java の ThreadLocal がメモリ オーバーフロー (メモリ不足) を引き起こす理由は何ですか?

  • Entry は弱参照 WeakReference を実装します。 、および Entry のキー (つまり、ビジネス パーティによって定義された ThreadLocal クラス) は、弱い参照にラップされ、Entry キーとして使用されます。 Java の弱参照の定義は、JVM がガベージ コレクション スキャンを実行し、弱参照のみを持つオブジェクトを検出すると、そのオブジェクトをすぐにリサイクルするというものです。これは、ThreadLocal が最初に設計されたときのメモリ オーバーフローを防ぐ手段です。

ソース コードは次のとおりです:

Java の ThreadLocal がメモリ オーバーフロー (メモリ不足) を引き起こす理由は何ですか?

キーは弱い参照にパッケージ化されており、によってリサイクルされます。ガベージ コレクション メカニズム、値スレッド (Thread) が終了しない場合、強参照チェーンが存在する可能性があります。

value は強参照であるため、Thread が存在する限り、スレッド プールなど、 は消滅しません。この強力な参照チェーンは存在しますが、value はリサイクルされず、メモリ オーバーフローが発生する可能性があります。

Thread ==> ThreadLocalMap == > Entry ==> value

ただし、この強参照チェーンを削除するアクションは、ケースのビジネス パーティによってトリガーされる必要があります。おそらくビジネス パーティが get の方法を知らないか、get のキーが空ではない可能性があり、 expungeStaleEntry クラスはトリガーされません。したがって、開発者は良い習慣を身につけ、終了したら ThreadLocal.remove()

メソッドまたは

ThreadLocal.set(null)# を忘れずに呼び出す必要があります。

public class ThreadLocalTest {
    /**
     * 未初始化的本地线程变量
     */
    private static ThreadLocal<User> userThreadLocal = new ThreadLocal<>();
    public static void main(String[] args) throws InterruptedException {
        int threadNum = 10;
        List<Thread> threadList = Lists.newArrayList();

        for (int i = 0; i < threadNum; ++i) {
            long userId = i;
            Thread t = new Thread(() -> {
                try {
                    // 设置变量值
                    userThreadLocal.set(new User(userId, "lanxing" + userId, "2x"));
                    // 使用变量
                    doSomething();
                } finally {
                    // 移除变量
                    userThreadLocal.remove();   //移除ThreadLocal变量
                }
            }, "T" + i);
            threadList.add(t);
            t.start();
        }

        for (int i = 0; i < threadNum; ++i) {
            threadList.get(i).join();
        }
    }
    private static void doSomething() {
        log.info("Use ThreadLocal variable :{}", JSON.toJSONString(userThreadLocal.get()));
    }
}
@Data
@NoArgsConstructor
@AllArgsConstructor
class User {
    private Long id;
    private String name;
    private String level;
}
印刷結果:

14:30:26.790 [T2] 情報 io.zhengsh.order.tool .ThreadLocalTest - ThreadLocal 変数を使用します:{" id":2,"level":"2x","name":"lanxing2"}

14:30:26.789 [T5] INFO io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{"id ":5,"level":"2x","name":"lanxing5"}

14:30:26.792 [T0] 情報 io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{"id" :0,"level":"2x","name":"lanxing0"}14:30:26.792 [T4] 情報 io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{"id": 4,"level":"2x","name":"lanxing4"}14:30:26.792 [T8] 情報 io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{"id":8 ,"level":"2x","name":"lanxing8"}

14:30:26.791 [T1] 情報 io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{"id":1, "level":"2x","name":"lanxing1"}
14:30:26.792 [T7] 情報 io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{"id":7," level":"2x","name":"lanxing7"}

14:30:26.792 [T6] INFO io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します: {"id":6,"level ":"2x","name":"lanxing6"}
14:30:26.791 [T9] 情報 io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{ "id":9,"level" :"2x","name":"lanxing9"}
14:30:26.790 [T3] 情報 io.zhengsh.order.tool.ThreadLocalTest - ThreadLocal 変数を使用します:{" id":3,"level": "2x","name":"lanxing3"}

以上がJava の ThreadLocal がメモリ オーバーフロー (メモリ不足) を引き起こす理由は何ですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はyisu.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。