首先来了解一下什么是内存泄漏:内存泄漏就是说我们在为一个变量分配内存的时候,在使用完成以后未及时的对这块内存进行清理,造成失去了对这块内存的一个控制
先看一段代码
public class ThreadLocalOutOfMemoryTest {
static class LocalVariable {
private Long[] a = new Long[1024*1024];
}
// (1)
final static ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(6, 6, 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");
}
}
先对代码进行一个讲解,我们定义了一个线程池,放置了6个线程,然后放置了50个任务,每个任务往localVariable中放置一个对象,那再引出今天的话题为什么会出现内存泄漏,首先我们在main函数中未使用shutdown对线程池进行退出,那么也就意味着这些线程会一直存在,jvm也不会退出
情况一:未使用localVariable.remove(),那这样的话,当在每个线程中使用了localVariable.set(new LocalVariable())时,未及时进行释放,jvm又不会退出,那么这个设置的值就会一直存在在内存中,不会被释放
情况二:使用了localVariable.remove(),那么,每次使用完线程,都会对当前线程内存进行一个清理和处理,就不会出现内存泄露的问题了
注:假如在实际项目中我们在线程池中通过Threadlocal设置了值,一定要在使用完成之后进行及时的清理,否则,只要程序不退出。这个设置的值会一直占用在内存中无法释放