Heim  >  Artikel  >  Java  >  Wie verwende ich die ThreadLocal-Klasse in Java?

Wie verwende ich die ThreadLocal-Klasse in Java?

王林
王林nach vorne
2023-05-08 23:49:061179Durchsuche

    Was ist der Nutzen von Threadlocal:

    Einfach ausgedrückt wird ein ThreadLocal in einem Thread geteilt und kann sein Wird in verschiedenen Threads verwendet. Threads sind voneinander isoliert (jeder Thread kann nur den Wert seines eigenen Threads sehen). Wie unten gezeigt:

    Wie verwende ich die ThreadLocal-Klasse in Java?

    ThreadLocal-Verwendungsbeispiel

    API-Einführung

    Bevor wir Threadlocal verwenden, haben wir Werfen Sie zunächst einen Blick auf die API unten:

    Wie verwende ich die ThreadLocal-Klasse in Java?

    Die API der ThreadLocal-Klasse ist sehr einfach. Die wichtigeren hier sind get(), set(. ), Remove( ), Set wird für Zuweisungsvorgänge verwendet, Get wird zum Abrufen des Werts einer Variablen verwendet und Remove dient zum Löschen des Werts der aktuellen Variablen. Es ist zu beachten, dass die Methode initialValue ausgelöst wird, wenn dies der Fall ist Wird zum ersten Mal aufgerufen und wird zum Initialisieren des aktuellen Variablenwerts verwendet. Standardmäßig gibt initialValue null zurück.

    Verwendung von ThreadLocal

    Nachdem wir mit dem Gespräch über die API der ThreadLocal-Klasse fertig sind, üben wir es, um den vorherigen Satz zu verstehen: Ein ThreadLocal befindet sich in einem Thread gemeinsam genutzt und isoliert zwischen verschiedenen Threads (jeder Thread kann nur den Wert seines eigenen Threads sehen)

    public class ThreadLocalTest {
    
        private static ThreadLocal<Integer> threadLocal = new ThreadLocal<Integer>() {
    	// 重写这个方法,可以修改“线程变量”的初始值,默认是null
            @Override
            protected Integer initialValue() {
                return 0;
            }
        };
    
        public static void main(String[] args) throws InterruptedException {
    
            //一号线程
            new Thread(new Runnable() {
                @Override
                public void run() {
                    System.out.println("一号线程set前:" + threadLocal.get());
                    threadLocal.set(1);
                    System.out.println("一号线程set后:" + threadLocal.get());
                }
            }).start();
    
            //二号线程
            new Thread(new Runnable() {
                @Override
                public void run() {
                    System.out.println("二号线程set前:" + threadLocal.get());
                    threadLocal.set(2);
                    System.out.println("二号线程set后:" + threadLocal.get());
    
                }
            }).start();
    
            //主线程睡1s
            Thread.sleep(1000);
    
            //主线程
            System.out.println("主线程的threadlocal值:" + threadLocal.get());
    
        }
    
    }

    Erklären Sie den obigen Code ein wenig:

    #🎜🎜 #Jede ThreadLocal-Instanz ähnelt a Variablennamen. Verschiedene ThreadLocal-Instanzen enthalten einen Wert (vorerst so verstanden), der in der folgenden Beschreibung als „ThreadLocal-Variable“ bezeichnet wird .

    Erstellen Sie eine statische „ThreadLocal-Variable“ in der Klasse, erstellen Sie zwei Threads im Hauptthread und setzen Sie die ThreadLocal-Variablen in diesen beiden Threads auf 1 bzw. 2. Warten Sie dann, bis die Ausführung der Threads Nr. 1 und 2 abgeschlossen ist, und überprüfen Sie den Wert der ThreadLocal-Variablen im Hauptthread.

    Programmergebnisse und Analyse⌛

    Wie verwende ich die ThreadLocal-Klasse in Java?

    Der Fokus der Programmergebnisse liegt darauf, dass die Ausgabe des Hauptthreads 0 ist. Wenn es sich um eine gewöhnliche Variable handelt, setzen Sie gewöhnliche Variablen in Thread 1 und 2 auf 1 und 2 und drucken Sie diese Variable dann aus, nachdem Thread 1 und 2 die Ausführung abgeschlossen haben. Der Ausgabewert muss 1 oder 2 sein (welcher ausgegeben wird, wird durch den Betrieb bestimmt). System im Zusammenhang mit der Thread-Planungslogik). Nach Verwendung der ThreadLocal-Variablen zum Zuweisen von Werten über zwei Threads wird jedoch der Anfangswert 0 im Hauptthread-Thread ausgegeben. Aus diesem Grund wird „ein ThreadLocal in einem Thread gemeinsam genutzt und von verschiedenen Threads isoliert“. Dies ist die Kernaufgabe von ThreadLocal: die Implementierung lokaler Variablen.

    Quellcode-Analyse von Threadlocal

    Prinzip

    Jedes Thread-Objekt verfügt über eine ThreadLocalMap. Wenn ein ThreadLocal erstellt wird, werden die ThreadLocal-Objekte hinzugefügt zur Map, wobei der Schlüssel ThreadLocal ist und der Wert einen beliebigen Typ haben kann. Möglicherweise verstehen Sie diesen Satz beim ersten Lesen nicht sehr gut, aber wir werden ihn verstehen, nachdem wir gemeinsam den Quellcode gelesen haben.

    Unser bisheriges Verständnis war, dass alle konstanten Werte oder Referenztypen in ThreadLocal-Instanzen gespeichert sind, aber tatsächlich ist dies nicht der Fall. Diese Aussage ermöglicht uns lediglich, dieses Konzept besser zu verstehen. Durch das Speichern eines Werts in ThreadLocal wird tatsächlich ein Wert in der ThreadLocalMap im aktuellen Thread-Objekt gespeichert. ThreadLocalMap kann einfach als Map verstanden werden, und der Schlüssel zum Speichern eines Werts in dieser Map ist die ThreadLocal-Instanz selbst.

    Quellcode

    Wie verwende ich die ThreadLocal-Klasse in Java?

    ???? Mit anderen Worten, die Daten in ThreadLocal, die Sie speichern möchten, sind tatsächlich nicht vorhanden Anstatt es in einem ThreadLocal-Objekt zu speichern, verwendet es diese ThreadLocal-Instanz als Schlüssel, um es in einer Map im aktuellen Thread zu speichern. Das Gleiche gilt, wenn der Wert von ThreadLocal abgerufen wird. Aus diesem Grund kann ThreadLocal eine Isolierung zwischen Threads erreichen.

    Interne Klasse ThreadLocalMap

    ThreadLocalMap ist eine interne Klasse von ThreadLocal, die einen Satz ihrer eigenen Map-Struktur implementiert✨

    ThreadLocalMap-Attribut: #🎜 🎜#
    static class Entry extends WeakReference<ThreadLocal<?>> {
                Object value;
                Entry(ThreadLocal<?> k, Object v) {
                    super(k);
                    value = v;
                }
            }
            //初始容量16
            private static final int INITIAL_CAPACITY = 16;
            //散列表
            private Entry[] table;
            //entry 有效数量 
            private int size = 0;
            //负载因子
            private int threshold;

    ThreadLocalMap setzt die ThreadLocal-Variable

    private void set(ThreadLocal<?> key, Object value) {
                Entry[] tab = table;
                int len = tab.length;
                
                //与运算  & (len-1) 这就是为什么 要求数组len 要求2的n次幂 
                //因为len减一后最后一个bit是1 与运算计算出来的数值下标 能保证全覆盖 
                //否者数组有效位会减半 
                //如果是hashmap 计算完下标后 会增加链表 或红黑树的查找计算量 
                int i = key.threadLocalHashCode & (len-1);
                
                // 从下标位置开始向后循环搜索  不会死循环  有扩容因子 必定有空余槽点
                for (Entry e = tab[i];   e != null;  e = tab[i = nextIndex(i, len)]) {
                    ThreadLocal<?> k = e.get();
                    //一种情况 是当前引用 返回值
                    if (k == key) {
                        e.value = value;
                        return;
                    }
                    //槽点被GC掉 重设状态 
                    if (k == null) {
                        replaceStaleEntry(key, value, i);
                        return;
                    }
                }
    			//槽点为空 设置value
                tab[i] = new Entry(key, value);
                //设置ThreadLocal数量
                int sz = ++size;
    			
    			//没有可清理的槽点 并且数量大于负载因子 rehash
                if (!cleanSomeSlots(i, sz) && sz >= threshold)
                    rehash();
            }

    ThreadLocalMap-Attributeinführung????:

      wird in einem Array gespeichert Ähnlich wie eine normale Hashmap Within, aber im Gegensatz zur Zipper-Methode, die Hashmap zur Lösung von Hash-Konflikten verwendet, verwendet ThreadLocalMap die Open-Adress-Methode
    • Die anfängliche Kapazität des Arrays beträgt 16 , und der Lastfaktor beträgt 2/3#🎜🎜 #
    • Der Schlüssel des Knotenknotens kapselt WeakReference für das Recycling
    • #🎜🎜 #ThreadLocalMap-Speicherort

    • In Thread gespeichert, gibt es zwei ThreadLocalMap-Variablen

    threadLocals werden im ThreadLocal-Objektmethodensatz erstellt und werden auch von ThreadLocal verwaltet

    public void set(T value) {
            Thread t = Thread.currentThread();
            ThreadLocalMap map = getMap(t);
            if (map != null)
                map.set(this, value);
            else
                createMap(t, value);
        }
    
        void createMap(Thread t, T firstValue) {
            t.threadLocals = new ThreadLocalMap(this, firstValue);
        }
    # 🎜🎜#inheritableThreadLocals Ähnlich wie ThreadLocal überschreibt InheritableThreadLocal die Methode createMapWie verwende ich die ThreadLocal-Klasse in Java?
    void createMap(Thread t, T firstValue) {
            t.inheritableThreadLocals = new ThreadLocalMap(this, firstValue);
        }

    inheritableThreadLocals 作用是将ThreadLocalMap传递给子线程

    Wie verwende ich die ThreadLocal-Klasse in Java?

    init方法中 条件满足后直接为子线程创建ThreadLocalMap

    Wie verwende ich die ThreadLocal-Klasse in Java?

    注意:

    • 仅在初始化子线程的时候会传递 中途改变副线程的inheritableThreadLocals 变量 不会将影响结果传递到子线程 。

    • 使用线程池要注意 线程不回收 尽量避免使用父线程的inheritableThreadLocals 导致错误

    Key的弱引用问题

    为什么要用弱引用,官方是这样回答的

    To help deal with very large and long-lived usages, the hash table entries use WeakReferences for keys.

    为了处理非常大和生命周期非常长的线程,哈希表使用弱引用作为 key。

    生命周期长的线程可以理解为:线程池的核心线程

    ThreadLocal在没有外部对象强引用时如Thread,发生GC时弱引用Key会被回收,而Value是强引用不会回收,如果创建ThreadLocal的线程一直持续运行如线程池中的线程,那么这个Entry对象中的value就有可能一直得不到回收,发生内存泄露。

    • key 使用强引用????: 引用的ThreadLocal的对象被回收了,但是ThreadLocalMap还持有ThreadLocal的强引用,如果没有手动删除,ThreadLocal不会被回收,导致Entry内存泄漏。

    • key 使用弱引用????: 引用的ThreadLocal的对象被回收了,由于ThreadLocalMap持有ThreadLocal的弱引用,即使没有手动删除,ThreadLocal也会被回收。value在下一次ThreadLocalMap调用set,get,remove的时候会被清除。

    Java8中已经做了一些优化如,在ThreadLocal的get()、set()、remove()方法调用的时候会清除掉线程ThreadLocalMap中所有Entry中Key为null的Value,并将整个Entry设置为null,利于下次内存回收。

    java中的四种引用

    • 强引用????: 如果一个对象具有强引用,它就不会被垃圾回收器回收。即使当前内存空间不足,JVM也不会回收它,而是抛出 OutOfMemoryError 错误,使程序异常终止。如果想中断强引用和某个对象之间的关联,可以显式地将引用赋值为null,这样一来的话,JVM在合适的时间就会回收该对象

    • 软引用????: 在使用软引用时,如果内存的空间足够,软引用就能继续被使用,而不会被垃圾回收器回收,只有在内存不足时,软引用才会被垃圾回收器回收。(软引用可用来实现内存敏感的高速缓存,比如网页缓存、图片缓存等。使用软引用能防止内存泄露,增强程序的健壮性)

    • 弱引用????: 具有弱引用的对象拥有的生命周期更短暂。因为当 JVM 进行垃圾回收,一旦发现弱引用对象,无论当前内存空间是否充足,都会将弱引用回收。不过由于垃圾回收器是一个优先级较低的线程,所以并不一定能迅速发现弱引用对象

    • 虚引用????: 虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。虚引用必须和引用队列(ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。(注意哦,其它引用是被JVM回收后才被传入ReferenceQueue中的。由于这个机制,所以虚引用大多被用于引用销毁前的处理工作。可以使用在对象销毁前的一些操作,比如说资源释放等。)

    通常ThreadLocalMap的生命周期跟Thread(注意线程池中的Thread)一样长,如果没有手动删除对应key(线程使用结束归还给线程池了,其中的KV不再被使用但又不会GC回收,可认为是内存泄漏),一定会导致内存泄漏,但是使用弱引用可以多一层保障:弱引用ThreadLocal会被GC回收,不会内存泄漏,对应的value在下一次ThreadLocalMap调用set,get,remove的时候会被清除,Java8已经做了上面的代码优化。

    Das obige ist der detaillierte Inhalt vonWie verwende ich die ThreadLocal-Klasse in Java?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

    Stellungnahme:
    Dieser Artikel ist reproduziert unter:yisu.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen