What is a ThreadLocal
ThreadLocalMap is a structure used to keep variables in a Thread from being disturbed by other threads. It contains a ThreadLocalMap class, which is a local variable of Thread class. The Map stores the key of the ThreadLocal object and the value of the object we want to store. In different threads, the variables held are actually copies of the current thread, completely isolated from other threads, in order to ensure that the thread execution process is not affected by other threads.
There are four main methods
1. Void set(Object value) Sets the thread-local variable value of the current thread. 2. Public Object get() This method returns the thread-local variable corresponding to the current thread. 3. Public void remove() removes the value of a thread-local variable. This method is new in JDK 5.0. It is important to note that local variables to the thread are automatically garbage collected when the thread terminates, so it is not necessary to explicitly call this method to clean up local variables of the thread, but it can speed up memory collection. 4. Protected Object initialValue() returns the initialValue of the thread-local variable. This method is a protected method, obviously designed to be overridden by subclasses. This method is a deferred call that is executed only once, the first time the thread calls GET () or set(Object). The default implementation in ThreadLocal simply returns null.
It’s worth noting that in JDK5.0, ThreadLocal already supports generics, and the class name has changed to ThreadLocal. The new API methods are void set(T value), T get(), and T initialValue().
Application scenarios of ThreadLocal
1. It is convenient for the same thread to use an object and avoid unnecessary parameter transfer; 2. Data isolation between threads (each thread uses its own local variables in its own thread, and ThreadLocal objects between threads do not affect each other); 3. Get database connection, Session, association ID (such as uniqueID of log, convenient to string up multiple logs);
Memory leak related problems
The primary cause of memory leaks is the design of entries in ThreadLocalMap, its internal class.
WeakReference<ThreadLocal<? >>, that is, the key of the Entry is a weak reference, so the key’ will be reclaimed in garbage collection, but the value corresponding to the key will not be reclaimed. In this case, the key is null and the value has a value.
If the key is empty, the value is invalid, and over time, the value accumulation will lead to memory leaks.
How to fix the memory leak
ThreadLocal calls its remove() method to clean up data each time it is used. Because its remove method will actively remove the current key and value(Entry).
Application scenario code examples
package com.zhang.mooc.juc.a02.threadlocal;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/ * * * ThreadLocalNormalUsage05: using ThreadLocal, assign each thread its own dateFormat object, guarantee the thread safe, efficient use of memory * *@author zhangxiaoxiang
* @date2020/7/26 * /
public class ThreadLocalNormalUsage05 {
// Create a thread pool.
public static ExecutorService threadPool = Executors.newFixedThreadPool(10);
public static void main(String[] args) throws InterruptedException {
for (int i = 0; i < 1000; i++) {
int finalI = i;
// Submit a Runnable task for execution and return a Future representing the task
threadPool.submit(new Runnable() {
@Override
public void run(a) {
String date = newThreadLocalNormalUsage05().date(finalI); System.out.println(date); }}); }// Start an orderly shutdown in the execution of previously submitted tasks, but no new tasks will be accepted. The call has no additional effect if already closed.
// This method does not wait for previously committed tasks to complete execution. Use awaitTermination to do this.
threadPool.shutdown();
}
public String date(int seconds) {
// The parameter is in milliseconds and is timed from 1970.1.1 00:00:00 GMT (1970.1.1 08:00:00 GMT)
Date date = new Date(1000 * seconds);
// get(): returns the value in the current thread copy of this thread-local variable
SimpleDateFormat dateFormat = ThreadSafeFormatter.dateFormatThreadLocal2.get();
returndateFormat.format(date); }}class ThreadSafeFormatter {
public static ThreadLocal<SimpleDateFormat> dateFormatThreadLocal = new ThreadLocal<SimpleDateFormat>() {
//ThreadLocal returns the initial value of the current thread as the thread-local variable
@Override
protected SimpleDateFormat initialValue(a) {
return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); }};/ / JDK8 new writing
public static ThreadLocal<SimpleDateFormat> dateFormatThreadLocal2 = ThreadLocal
.withInitial(() -> new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));
}
Copy the code