Introduction to the
The ThreadLocal class is used to provide local variables within a thread. This allows variables to be accessed in a multithreaded environment (through get and SET methods) to ensure that variables in each thread are relatively independent of variables in other threads. ThreadLocal instances are typically of private static type and are used to associate threads with thread contexts.
ThreadLocal provides local variables within a thread that do not interfere with each other. These variables operate for the lifetime of the thread and reduce the complexity of passing common variables between functions or components within the same thread.
conclusion
- Thread concurrency: In the case of multi-threaded concurrency
- Passing data: We can
ThreadLocal
Pass common variables in different components on the same thread - Thread isolation: Variables for each thread are independent and do not affect each other
The basic use
1. Common methods
Method statement | describe |
---|---|
ThreadLocal() | Create a ThreadLocal object |
public void set(T value) | Sets the local variable of the current thread binding |
public T get() | Gets the local variable of the current thread binding |
Public void remove() | Removes a local variable from the current thread binding |
2. Use cases
public class ThreadLocal_Demo_01 {
private String content;
public String getContent(a) {
return content;
}
public void setContent(String content) {
this.content = content;
}
public static void main(String[] args) {
ThreadLocal_Demo_01 demo_01 = new ThreadLocal_Demo_01();
for (int i = 0; i < 5; i++) {
Thread thread = new Thread(() -> {
demo_01.setContent(Thread.currentThread().getName() + "The data");
System.out.println("-- -- -- -- -- -- -- -- --");
System.out.println(Thread.currentThread().getName() + "-- -- -- -- >" + demo_01.getContent());
});
thread.setName("Thread"+i); thread.start(); }}}Copy the code
The results
The result shows that multiple threads fail to access the same variable, data between threads is not isolated, and ThreadLocal is used to resolve the problem
public class ThreadLocal_Demo_02 {
private static ThreadLocal<String> tl = new ThreadLocal<>();
public String getContent(a) {
return tl.get();
}
public void setContent(String content) {
tl.set(content);
}
public static void main(String[] args) {
ThreadLocal_Demo_02 demo_02 = new ThreadLocal_Demo_02();
for (int i = 0; i < 5; i++) {
Thread thread = new Thread(() -> {
demo_02.setContent(Thread.currentThread().getName() + "The data");
System.out.println("-- -- -- -- -- -- -- -- --");
System.out.println(Thread.currentThread().getName() + "-- -- -- -- >" + demo_02.getContent());
});
thread.setName("Thread"+ i); thread.start(); }}}Copy the code
Print the result
As a result, ThreadLocal provides a convenient solution to the problem of data isolation between multiple threads
3, the difference between ThreadLocal and synchronized keyword
The synchronized keyword can also be used to solve the concurrency problem
for (int i = 0; i < 5; i++) {
Thread thread = new Thread(() -> {
synchronized (ThreadLocal_Demo_01.class) {
demo_01.setContent(Thread.currentThread().getName() + "The data");
System.out.println("-- -- -- -- -- -- -- -- --");
System.out.println(Thread.currentThread().getName() + "-- -- -- -- >"+ demo_01.getContent()); }}); thread.setName("Thread"+ i); thread.start(); }}Copy the code
In dealing with multi-threaded concurrency safety, the most common method is to use locks to control the access of different threads to the critical region. However, any type of lock, optimistic lock or pessimistic lock, will have a certain impact on performance when concurrency conflicts.
ThreadLocal offers a different approach to thread-safety, in that rather than resolving conflicts when they occur, it avoids them altogether
The difference between
The ThreadLocal and sychronized keywords are both used to deal with the problem of multiple threads accessing variables concurrently, but they approach the problem from different perspectives.
sychronized | ThreadLocal | |
---|---|---|
The principle of | The synchronization mechanism uses a “time for space” approach, providing only one copy of a variable that different threads queue up to access | ThreadLocal Using the “space for time” method, each thread is provided with a copy of the variable, so as to achieve simultaneous access without interference |
focus | Synchronization of access to resources between multiple threads | In multithreading, separate data from each other between threads |
conclusion
In this case, ThreadLocal and sychronized are both effective, but ThreadLocal is more appropriate because it allows for higher concurrency
The benefits of ThreadLocal
1. Transfer data: Save the data guaranteed by each thread, which can be directly obtained where needed, avoiding the code coupling problem brought by direct transfer of parameters
2. Thread isolation: data between threads are isolated from each other but have concurrency to avoid performance loss caused by synchronous methods
Internal structure of ThreadLocal
JDK1.8 before
Each ThreadLocal creates a Map, uses the thread as the Map key, and stores local variables as the Map value, thus achieving the effect of local variable isolation for each thread.
After JDK1.8
The design of ThreadLocal in JDK8 is that each Thread maintains a ThreadLocalMap. The Map’s key is the ThreadLocal instance itself, and the value is the actual value to store Object.
The specific process is as follows:
-
Each Thread has an internal Map (ThreadLocalMap)
-
The Map stores the ThreadLocal object (key) and the thread variable copy (value).
-
The Map inside a Thread is maintained by a ThreadLocal, which is responsible for fetching and setting Thread variable values from the Map
-
For different threads, each time the copy value is obtained, other threads can not obtain the copy value of the current thread, forming the isolation of copies, mutual interference
Benefits of Design
1. After this design, each Map stores fewer entries. Because the number of stores was previously determined by the number of threads, it is now determined by the number of ThreadLocal. In practice, the number of ThreadLocal is usually less than the number of Thread
2. After a Thread is destroyed, the corresponding ThreadLocalMap is also destroyed, reducing memory usage
Thread isolation in ThreadLocal
public class Thread {
/* * ThreadLocal values pertaining to this thread. This map is maintained * by the ThreadLocal class. */
ThreadLocal.ThreadLocalMap threadLocals = null;
/* * InheritableThreadLocal values pertaining to this thread. This map is * maintained by the InheritableThreadLocal class. */
ThreadLocal.ThreadLocalMap inheritableThreadLocals = null;
}
Copy the code
Threadlocals and inheritableThreadLocals are two properties of the same Thread, so threadlocales for each Thread are isolated and exclusive. In the init method of a Thread, when the parent Thread creates the child Thread, the value of inheritableThreadLocals is copied, but not the value of threadlocals
// Init method in thread
private void init(ThreadGroup g, Runnable target, String name,
long stackSize, AccessControlContext acc) {...// When the inheritableThreadLocals value of the parent thread is not null
// All values in inheritable are passed to child threads
if(parent.inheritableThreadLocals ! =null)
this.inheritableThreadLocals =
// ThreadLocal.createInheritedMap(parent.inheritableThreadLocals); . }//
static ThreadLocalMap createInheritedMap(ThreadLocalMap parentMap) {
return new ThreadLocalMap(parentMap);
}
Copy the code
The core method of ThreadLocal
1. Set method
/** * Sets the ThreadLocal value of the current thread **@paramValue The value */ to be stored in the corresponding ThreadLocal of the current thread
public void set(T value) {
// Get the current thread
Thread t = Thread.currentThread();
// Get the ThreadLocalMap object maintained in this thread object
ThreadLocalMap map = getMap(t);
// Check whether the map exists
if(map ! =null)
// If it exists, call map.set to set entry for this entity
map.set(this, value);
else
ThreadLocalMap does not exist in the current Thread
// createMap is called to initialize ThreadLocalMap
// add t(the current thread) and value(the value of t) as the first entry to the ThreadLocalMap object
createMap(t, value);
}
/** * Get the ThreadLocalMap maintained by the current Thread **@paramValue The value */ to be stored in the corresponding ThreadLocal of the current thread
ThreadLocalMap getMap(Thread t) {
return t.threadLocals;
}
/** * Create ThreadLocalMap ** for the current Thread@paramT Current thread *@paramFirstValue Specifies the value of the first entry stored in the map */
void createMap(Thread t, T firstValue) {
// This is the threadLocal that calls this method
t.threadLocals = new ThreadLocalMap(this, firstValue);
}
/** * sets the value of the threadLocal association */
private void set(ThreadLocal
key, Object value) {
Entry[] tab = table;
int len = tab.length;
// Find the index location
int i = key.threadLocalHashCode & (len-1);
for(Entry e = tab[i]; e ! =null;
// Next index locatione = tab[i = nextIndex(i, len)]) { ThreadLocal<? > k = e.get();// If the keys are the same, replace value
if (k == key) {
e.value = value;
return;
}
// If the key is null, replace the old entry
if (k == null) {
replaceStaleEntry(key, value, i);
return;
}
}
tab[i] = new Entry(key, value);
int sz = ++size;
if(! cleanSomeSlots(i, sz) && sz >= threshold) rehash(); }private void replaceStaleEntry(ThreadLocal<? > key, Object value,int staleSlot) {
Entry[] tab = table;
intlen = tab.length; Entry e; .for (inti = nextIndex(staleSlot, len); (e = tab[i]) ! =null; i = nextIndex(i, len)) { ThreadLocal<? > k = e.get();// If we find key, then we need to swap it
// with the stale entry to maintain hash table order.
// The newly stale slot, or any other stale slot
// encountered above it, can then be sent to expungeStaleEntry
// to remove or rehash all of the other entries in run.
if (k == key) {
e.value = value;
tab[i] = tab[staleSlot];
tab[staleSlot] = e;
// Start expunge at preceding stale entry if it exists
if (slotToExpunge == staleSlot)
slotToExpunge = i;
cleanSomeSlots(expungeStaleEntry(slotToExpunge), len);
return; }... }/** * Remove an old entry by reprocessing any possible colliding entries lying between the old slot and the next empty slot */
private int expungeStaleEntry(int staleSlot) {
Entry[] tab = table;
int len = tab.length;
/ / delete
tab[staleSlot].value = null;
tab[staleSlot] = null;
size--;
// Rehash until we encounter null
Entry e;
int i;
for(i = nextIndex(staleSlot, len); (e = tab[i]) ! =null; i = nextIndex(i, len)) { ThreadLocal<? > k = e.get();if (k == null) {
e.value = null;
tab[i] = null;
size--;
} else {
int h = k.threadLocalHashCode & (len - 1);
if(h ! = i) { tab[i] =null;
// Unlike Knuth 6.4 Algorithm R, we must scan until
// null because multiple entries could have been stale.
while(tab[h] ! =null) h = nextIndex(h, len); tab[h] = e; }}}return i;
}
Copy the code
Code execution flow
- First get the current thread, and get a Map based on the current thread
- If the Map obtained is not empty, the parameter is set to the Map (the reference to the current threadLocal is the key)
- If the Map is empty, the Map is created for the thread and the initial value is set
2. Get method
/** * returns the value of the ThreadLocal stored in the current thread.@linkThe #initialValue} method initializes the value * *@returnReturns the value */ of this ThreadLocal for the current thread
public T get(a) {
// Get the current thread
Thread t = Thread.currentThread();
// Get the ThreadLocal object maintained in this thread object
ThreadLocalMap map = getMap(t);
// If the map exists
if(map ! =null) {
// With the current ThreadLocal as the key, call getEntry to get the corresponding store entry
ThreadLocalMap.Entry e = map.getEntry(this);
// void e
if(e ! =null) {
@SuppressWarnings("unchecked")
// Get the value of the storage entity e
// is the value of the ThreadLocal that we want for the current thread
T result = (T)e.value;
returnresult; }}/ / initialization
// The map does not exist, indicating that the thread has no ThreadLocal object to maintain
// The map exists, but there is no entry associated with the current ThreadLocal
return setInitialValue();
}
/** * initializes */
private T setInitialValue(a) {
// Call initalValue to get the initialized value
// This method can be overridden by subclasses, otherwise null is returned by default
T value = initialValue();
// Get the current thread object
Thread t = Thread.currentThread();
// Get the ThreadLocalMap object maintained in this thread object
ThreadLocalMap map = getMap(t);
if(map ! =null)
// If it exists, call map.set to set entry for this entity
map.set(this, value);
else
// 1) The current Thread does not have a ThreadLocalMap object
// 2) createMap is called to initialize ThreadLocalMap
// 3) store t(the current thread) and value(the corresponding value of t) as the first entry into ThreadLocalMap
createMap(t, value);
return value;
}
Copy the code
Execute the process
- First get the current thread, and get a Map based on the current thread
- If the obtained Map is not empty, the Map uses the reference of ThreadLocal as the key to obtain the corresponding Entry in the Map. Otherwise, go to 4
- If entry is not null, return e.value, otherwise go to 4
- If the Map is empty or e is empty, the initialValue function is used to get the initialValue value and then a new Map is created using a reference to ThreadLocal and value as the firstKey and firstValue
3, remove method
/** * Delete entry */ from ThreadLocal
public void remove(a) {
// Get the ThreadLocalMap object maintained in the current thread object
ThreadLocalMap m = getMap(Thread.currentThread());
if(m ! =null)
// If the Map exists, the remove method is called
// Delete the corresponding entity entry using the current ThreadLocal as the key
m.remove(this);
}
Copy the code
Execute the process
- Gets the current thread and a Map based on the current thread
- If the Map obtained is not empty, the entry corresponding to the current ThreadLocal is removed
4. InitialValue method
/** * returns the initial value of the ThreadLocal corresponding to the current thread ** The first call to this method occurs when a thread accesses the ThreadLocal value of this thread via the get method * unless the thread first calls the set method, in which case, InitalValue is not called by this thread * In general, This method is called at most once per thread * * this method simply returns null * if the programmer has an initial value other than null to the ThreadLocal ThreadLocal variable * the method must be overridden by subclass inheritance methods * */ can usually be implemented via anonymous inner classes * */
protected T initialValue(a){
return null;
}
Copy the code
Execute the process
- This method is a deferred call, as you can see from the code above, before the set method is called
get
Method, and only once - The default implementation of this method returns one directly
null
- You can override this method if you want to return an initial value other than NULL
As you can see, the underlying principles of the set, GET, and remove methods are relatively simple, but they all have one common feature: the use of ThreadLocalMap.
ThreadLocalMap source code analysis
ThreadLocalMap is a static inner class in ThreadLocal that is essentially a simple Map structure. Key is a ThreadLocal type, value is a ThreadLocal stored value, and the underlying data structure is an array of Entry types
The basic structure
Member variables
/** * Initial capacity */
private static final int INITIAL_CAPACITY = 16;
/** * The table where data is stored. The Entry class * must be an integer power of 2 */
private Entry[] table;
/** * The number of entrys in the array, which can be used to determine whether the current table usage exceeds the threshold */
private int size = 0;
/** * Specifies the capacity expansion threshold. If the table usage exceeds this threshold, capacity expansion will be performed */
private int threshold; // Default to 0
Copy the code
Storage structure -Entry
/** * Entry integrated WeakReference, If the key is null(entry.get() == null), it means that the key is not referenced * so the entry can also be cleared from the table */
static class Entry extends WeakReference<ThreadLocal<? >>{
/** The value associated with this ThreadLocal. */Object value; Entry(ThreadLocal<? > k, Object v) {super(k);
value = v;
}}
Copy the code
In ThreadLocalMap, Entry is also used to hold k-V structured data. However, the key in the Entry can only be a ThreadLocal object, which is specified in the constructor.
In addition, Entry inherits WeakReference, that is, key (ThreadLocal) is a WeakReference, whose purpose is to unbind the life cycle of ThreadLocal objects and the life cycle of threads.
Weak references and memory leaks
(1) Concepts related to memory leakage
There are two main cases of memory leakage: one is that the requested space in the heap is not released; The other is when the object is no longer in use, but is still in memory.
- Memory overflow: Indicates that there is not enough Memory for applicants to use.
- Memory leak: A Memory leak refers to a program that fails to release or release dynamically allocated heap Memory for some reason, resulting in a waste of system Memory, slowing down the program and even crashing the system. The accumulation of memory leaks eventually leads to an overflow of memory.
(2) Weak references to related concepts
There are four types of references in Java: strong, soft, weak, and virtual. The current issue involves strong and weak references:
A “Strong” Reference is the most common common object Reference. As long as there is a Strong Reference to an object, it indicates that the object is “alive” and the garbage collector will not reclaim it.
WeakReference: once the garbage collector finds an object with only weak references, it will reclaim its memory regardless of whether the current memory space is sufficient or not.
If the key uses weak references, ThreadLocal refs are reclaimed after ThreadLocal is used
Since ThreadLocalMap only holds weak references to ThreadLocal and no strong references to ThreadLocal instances, ThreadLocal can be reclaimed by GC with key=null in the Entry
CurrentThreadRef-> CurrentThread ->threadLocalMap -> Entry -> value The value is never collected, and the value is never accessed, resulting in a value memory leak
Weak references do not solve the problem of memory leaks. Instead, they are used to mark unwanted entries so that they can be scanned and set to NULL
If the key is null and the get/set method value is set to NULL, is remove necessary
The get and set methods may not be executed for a long time
Assume a strong reference
Assuming ThreadLocal is used up in the business code, ThreadLocal Ref is reclaimed.
However, because threadLocalMap entries strongly reference threadLocal, threadLocal cannot be reclaimed.
ThreadRef -> CurrentThread ->threadLocalMap-> Entry The Entry will not be reclaimed (the Entry contains ThreadLocal instances and values), resulting in an Entry memory leak.
That is, keys in ThreadLocalMap use strong references and are not completely immune to memory leaks.
Why is there a memory leak
Comparing the above two cases, we can see that the memory leak is independent of whether the key in ThreadLocalMap uses weak references. So what is the real cause of the memory leak?
You can see that in both of these memory leak cases, there are two premises
- The Entry was not manually deleted
- CurrentThread is still running
The first point is easy to understand. After using ThreadLocal, you can avoid memory leaks by calling the remove method to remove the corresponding Entry
Second, because ThreadLocalMap is an attribute of Thread and is referenced by the current Thread, it takes as long to declare as Thread. When ThreadLocal is used, if the current thread terminates, the ThreadLocalMap will also be collected by the GC, preventing a memory leak at the root.
In summary, the root cause of a ThreadLocal memory leak is that since threadLocalMaps have the same lifetime as threads, memory leaks can occur if the corresponding key is not manually removed.
Why use weak references
Through the root system, memory leaks are not completely avoided regardless of the type of reference used by the key in ThreadLocalMap, regardless of strong or weak references. So why use weak references
In fact, the set/getEntry method in ThreadLocalMap evaluates if the key is null (ThreadLocal is null), and if it is null, value is set to null.
This means that when ThreadLocal is used and currentThreads are still running, even if you forget to call the remove method, weak references are more secure than strong ones: Weak references to ThreadLocal are reclaimed, and the corresponding value is cleared the next time ThreadLocalMap calls any of the set, GET, or remove methods to avoid memory leaks.