8.2 Improper use of ThreadLocal may cause memory leaks

In this section, we will focus on the causes of memory leaks caused by using ThreadLocal, and explain the memory leaks caused by using ThreadLocal.

8.2.1 Why Do MEMORY Leaks Occur

ThreadLocals is a utility class that stores variables in the thread’s threadLocals variable, which is of type ThreadLocalMap,





image.png

As shown in the figure above, ThreadLocalMap is an Entry array, and Entry inherits WeakReference. The value inside an Entry is used to store the value passed by ThreadLocal’s set method. Then where is the ThreadLocal object itself stored? Let’s look at the Entry constructor:

Entry(ThreadLocal<? > k, Object v) { super(k); value = v; } public WeakReference(T referent) { super(referent); } Reference(T referent) { this(referent, null); } Reference(T referent, ReferenceQueue<? super T> queue) { this.referent = referent; this.queue = (queue == null) ? ReferenceQueue.NULL : queue; }Copy the code

It can be seen that k is passed to the constructor of WeakReference, that is to say, the key in ThreadLocalMap is the WeakReference of the ThreadLocal object, specifically, the referent variable references the ThreadLocal object. Value is the value passed by the set method that calls ThreadLocal specifically.

When a thread calls ThreadLocal’s set method to set a variable, the current thread’s ThreadLocalMap stores a record whose key is the reference to ThreadLocal and whose value is the set value. If the current thread exists without calling the remove method of ThreadLocal, and there are references to ThreadLocal elsewhere, The ThreadLocalMap variable of the current thread contains references to ThreadLocal variables and references to value objects that will not be released, causing a memory leak. But consider that if the ThreadLocal variable has no other strong dependencies and the current thread still exists, because the key in the thread’s ThreadLocalMap is weakly dependent, Weak references to ThreadLocal variables in the current thread’s ThreadLocalMap will be collected during GC, but the corresponding values will still leak memory. In this case, the ThreadLocalMap contains an entry with a null key but not a null value. There are times when a ThreadLocal set or get or remove method will clean up an entry with a null key, but this does not have to happen.

private void remove(ThreadLocal<? > key) {//(1) count the position of the current ThreadLocal variable in the table array, try to use the quick location method Entry[] TAB = table; int len = tab.length; int i = key.threadLocalHashCode & (len-1); Table array for (Entry e = TAB [I]; e ! = null; e = tab[i = nextIndex(i, (LLLLDB () == key) {//(LLLLDB () == key) {//(LLLLDB () == key) {//(LLLLDB () == key) {//(LLLLDB () == key); ExpungeStaleEntry (I); expungeStaleEntry(I); return; } } } private int expungeStaleEntry(int staleSlot) { Entry[] tab = table; int len = tab.length; TAB [staleSlot]. Value = null; tab[staleSlot] = null; size--; Entry e; int i; for (i = nextIndex(staleSlot, len); (e = tab[i]) ! = null; i = nextIndex(i, len)) { ThreadLocal<? > k = e.get(); //(7) If the key is null, the reference to value is removed. if (k == null) { e.value = null; tab[i] = null; size--; } else { int h = k.threadLocalHashCode & (len - 1); if (h ! = i) { tab[i] = null; while (tab[h] ! = null) h = nextIndex(h, len); tab[h] = e; } } } return i; }Copy the code

  • In step (4), Entry’s clear method is called. In fact, the clear method of parent WeakReference is called to remove the WeakReference to ThreadLocal.
  • Step (6) removes the reference to value, by which time the current ThreadLocal object in the current thread has been cleaned up.
  • Code (7) starts with the index of the current element to see if there are other elements in the table array whose key is null. The loop exits if it encounters a null element in the table. So we know that elements with a null key in the Entry following the null element will not be cleaned.

Conclusion: The key in the internal Entry of a ThreadLocalMap uses a weak reference to a ThreadLocal object. This is an improvement to avoid memory leaks, because if a strong reference is made, then even if there is no reference to a ThreadLocal object elsewhere, ThreadLocal objects in a ThreadLocalMap will not be reclaimed, and weak references will be reclaimed, even if the value of a ThreadLocalMap cannot be reclaimed. ThreadLocalMap provides set, GET, and remove methods to clean these entries in some cases, but this is not timely. This is not always done, so in some cases memory leaks will occur, so even calling the remove method after use is the best way to resolve memory leaks.

8.2.2 Memory Leaks caused by ThreadLocal in thread Pools

Let’s look at an example of using ThreadLocal in a thread pool:

public class ThreadPoolTest { static class LocalVariable { private Long[] a = new Long[1024*1024]; } // (1) final static ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(5, 5, 1, TimeUnit.MINUTES, new LinkedBlockingQueue<>()); // (2) final static ThreadLocal<LocalVariable> localVariable = new ThreadLocal<LocalVariable>(); public static void main(String[] args) throws InterruptedException { // (3) for (int i = 0; i < 50; ++i) { poolExecutor.execute(new Runnable() { public void run() { // (4) localVariable.set(new LocalVariable()); // (5) System.out.println("use local varaible"); //localVariable.remove(); }}); Thread.sleep(1000); } // (6) System.out.println("pool execute over"); }Copy the code

  • Code (1) creates a pool with a core number of threads and a maximum number of threads of 5. This ensures that there are 5 threads running in the pool at any one time.
  • Code (2) creates a ThreadLocal variable with a generic parameter LocalVariable that contains an array of longs.
  • Code (3) puts 50 tasks into the thread pool
  • Code (4) sets the localVariable of the current thread, i.e. putting the localVariable of new into the threadLocals variable of the current thread.
  • The user thread in the thread pool does not exit, and therefore the JVM process does not exit, because the shutdown or shutdownNow methods of the thread pool are not called.

Run the current code and use JConsole to monitor the heap memory changes as shown below:





image.png

Then unpack the localvariable.remove () comment, and then in run, observe the heap memory changes as follows:





image.png

It can be seen from the first running result that the process occupied about 77M memory when the main thread was asleep, and the second running result occupied about 25M memory. It can be seen that the memory was leaked when the code was running. The reasons for the leakage are analyzed below.

The localvariable.remove () method is not called after setting the localVariable of the thread. As a result, the new localVariable () instance in the threadLocals variable of the 5 threads in the thread pool is not released. The task in the thread pool completes, but the five threads in the thread pool remain until the JVM exits. It is important to note that localVariable is not reclaimed because it is declared static, even though there is a weak reference to localVariable in a thread’s ThreadLocalMap. The code running result 2 does not leak memory because the thread sets localVariable, even though it calls localvariable.remove () to clean it up.

The core thread in the thread pool will always exist. If not, the core thread of the thread pool will always hold the ThreadLocal variable in the threadLocals variable.

8.2.3 Memory Leaks Caused by Using ThreadLocal in Tomcat Servlets

Let’s start with the following code for a Servlet:

public class HelloWorldExample extends HttpServlet { private static final long serialVersionUID = 1L; static class LocalVariable { private Long[] a = new Long[1024 * 1024 * 100]; } //(1) final static ThreadLocal<LocalVariable> localVariable = new ThreadLocal<LocalVariable>(); @Override public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { //(2) localVariable.set(new LocalVariable()); response.setContentType("text/html"); PrintWriter out = response.getWriter(); out.println("<html>"); out.println("<head>"); out.println("<title>" + "title" + "</title>"); out.println("</head>"); out.println("<body bgcolor=\"white\">"); //(3) out.println(this.toString()); //(4) out.println(Thread.currentThread().toString()); out.println("</body>"); out.println("</html>"); }}Copy the code

  • Code (1) creates a localVariable object,
  • Code (2) sets the localVariable value in the servlet’s doGet method
  • Code (3) prints an instance of the current servlet
  • Code (4) prints the current thread

Sever. XML configuration in tomcat conf:

<Executor name="tomcatThreadPool" namePrefix="catalina-exec-" maxThreads="10" minSpareThreads="5"/> <Connector Executor ="tomcatThreadPool" port="8080" protocol="HTTP/1.1" connectionTimeout="20000" redirectPort="8443" />Copy the code

The maximum number of threads in the tomcat thread pool is 10 and the minimum number of threads is 5. Here is a review of Tomcat’s container structure, as shown below:




image.png

The Connector component of Tomcat is responsible for accepting and processing requests. Socket acceptor threads are responsible for accepting user access requests and then handing the received requests to the Worker Threads Pool for specific processing. The latter is the thread pool we configured in server.xml. The threads in the Worker Threads Pool distribute specific requests to specific application servlets for processing.

With that knowledge in mind, let’s start Tomcat and access the servlet multiple times and see that it is possible to output the following results

HelloWorldExample @ 2 a10b2d2 Thread/catalina - exec - 5, 5, the main HelloWorldExample @ 2 a10b2d2 Thread/catalina - exec - 1, 5, the main HelloWorldExample @ 2 a10b2d2 Thread/catalina - exec - 4, 5, the mainCopy the code

The first part is the printed servlet instance, which means that the servlet instance accessed for many times is the same. In the second part, catalina-exec-5, CATALina-exec-1, catalina-exec-4, Serlvet is executed using threads 5, 1, and 4 from the connector thread pool. If you open jConsole while accessing the servlet and observe the heap memory, you will find that the memory will spike because when the worker thread calls the servlet’s doGet method, the new LocalVariable() instance will be added to the worker thread’s threadLocals variable. However, if the servlet is not removed, it may be accessed multiple times using a different thread from the worker thread pool, resulting in memory leaks from multiple threads in the worker thread pool.

To make matters worse, in the tomcat6.0 era, applying a reload operation would cause the webappClassLoader to reload the application. The localvariable. class contains a reference to the webAppClassLoader. The localVariable. class contains a reference to the webAppClassLoader. Since the LocalVariable instance is not freed, the localVariable. class object is not freed, so the WebAppClassLoader is not freed, and all classes loaded by the WebAppClassLoader are not freed. This is because the worker thread pool of the Connector component is always available when reload is applied, and the threadLocals variable is not cleaned up. In tomcat7.0, this problem is fixed. The application will clean up the worker thread pool threadLocals variable when being reloaded.

December 31, 2017 afternoon 5:44:24 org. Apache. Catalina. Loader. WebappClassLoader checkThreadLocalMapForLeaks severe: The web application [/examples] created a ThreadLocal with key of type [java.lang.ThreadLocal] (value [java.lang.ThreadLocal@63a3e00b]) and a value of type [HelloWorldExample.LocalVariable] (value [HelloWorldExample$LocalVariable@4fd7564b]) but failed to remove it when the web application was stopped. Threads are going to be renewed over time to try and avoid a probable memory leak.Copy the code

8.2.4 summary

Java’s ThreadLocal is a great tool for programming, but it can be deadly if used incorrectly. Make it a rule to remove ThreadLocal variables when they are used in a thread.



Gardo: The Beauty of Concurrent Programming in Java is already on salezhuanlan.zhihu.com