memoryleaktestcase.java
来自「这是一个有关common beanutils 的源码」· Java 代码 · 共 575 行 · 第 1/2 页
JAVA
575 行
// Clear All BeanUtils caches after the test
clearAllBeanUtilsCaches();
}
/**
* Tests that WrapDynaClass's dynaClasses doesn't cause a memory leak.
*/
public void testWrapDynaClass_dynaClasses_memoryLeak() throws Exception {
if (isPre15JVM()) {
return;
}
// Clear All BeanUtils caches before the test
clearAllBeanUtilsCaches();
String className = "org.apache.commons.beanutils.memoryleaktests.pojotests.SomePojo";
// The classLoader will go away only when these following variables are released
ClassLoader loader = newClassLoader();
Class beanClass = loader.loadClass(className);
Object bean = beanClass.newInstance();
WrapDynaBean wrapDynaBean = new WrapDynaBean(bean);
// -----------------------------------------------------------------------------
WeakReference someRef = new WeakReference(loader);
// Sanity checks only
assertNotNull("ClassLoader is null", loader);
assertNotNull("BeanClass is null", beanClass);
assertNotSame("ClassLoaders should be different..", getClass().getClassLoader(), beanClass.getClassLoader());
assertSame("BeanClass ClassLoader incorrect", beanClass.getClassLoader(), loader);
// if you comment the following line, the testcase will work, and the ClassLoader will be released.
// That proves that nothing is wrong with the test, and WrapDynaClass is holding a reference
assertEquals("initialValue", wrapDynaBean.get("name"));
// this should make the reference go away.
loader = null;
beanClass = null;
bean = null;
wrapDynaBean = null;
// Wrap Dyna Class uses the PropertyUtilsBean's decriptor caches.
// Uncomment the following line to check this is not just a repeat of that memory leak.
// BeanUtilsBean.getInstance().getPropertyUtils().clearDescriptors();
forceGarbageCollection(); /* Try to force the garbage collector to run by filling up memory */
if (someRef.get() != null) {
profilerLeakReport("WrapDynaClass dynaClasses", className);
}
// if everything is fine, this will be null
assertNull("WrapDynaClass is holding a reference to the classLoader", someRef.get());
// Clear All BeanUtils caches after the test
clearAllBeanUtilsCaches();
}
/**
* Tests that ConvertUtilsBean's converters doesn't cause a memory leak.
*/
public void testConvertUtilsBean_converters_memoryLeak() throws Exception {
// Clear All BeanUtils caches before the test
clearAllBeanUtilsCaches();
String className = "org.apache.commons.beanutils.memoryleaktests.pojotests.CustomInteger";
// The classLoader will go away only when these following variables are released
ClassLoader loader = newClassLoader();
Class beanClass = loader.loadClass(className);
Object bean = beanClass.newInstance();
// -----------------------------------------------------------------------------
WeakReference someRef = new WeakReference(loader);
// Sanity checks only
assertNotNull("ClassLoader is null", loader);
assertNotNull("BeanClass is null", beanClass);
assertNotSame("ClassLoaders should be different..", getClass().getClassLoader(), beanClass.getClassLoader());
assertSame("BeanClass ClassLoader incorrect", beanClass.getClassLoader(), loader);
// if you comment the following two lines, the testcase will work, and the ClassLoader will be released.
// That proves that nothing is wrong with the test, and ConvertUtilsBean is holding a reference
ConvertUtils.register(new IntegerConverter(), beanClass);
assertEquals("12345", ConvertUtils.convert(bean, String.class));
// this should make the reference go away.
loader = null;
beanClass = null;
bean = null;
forceGarbageCollection(); /* Try to force the garbage collector to run by filling up memory */
if (someRef.get() != null) {
profilerLeakReport("ConvertUtilsBean converters", className);
}
// if everything is fine, this will be null
assertNull("ConvertUtilsBean is holding a reference to the classLoader", someRef.get());
// Clear All BeanUtils caches after the test
clearAllBeanUtilsCaches();
}
/**
* Tests that LocaleConvertUtilsBean's converters doesn't cause a memory leak.
*/
public void testLocaleConvertUtilsBean_converters_memoryLeak() throws Exception {
// Clear All BeanUtils caches before the test
clearAllBeanUtilsCaches();
String className = "org.apache.commons.beanutils.memoryleaktests.pojotests.CustomInteger";
// The classLoader will go away only when these following variables are released
ClassLoader loader = newClassLoader();
Class beanClass = loader.loadClass(className);
Object bean = beanClass.newInstance();
// -----------------------------------------------------------------------------
WeakReference someRef = new WeakReference(loader);
// Sanity checks only
assertNotNull("ClassLoader is null", loader);
assertNotNull("BeanClass is null", beanClass);
assertNotSame("ClassLoaders should be different..", getClass().getClassLoader(), beanClass.getClassLoader());
assertSame("BeanClass ClassLoader incorrect", beanClass.getClassLoader(), loader);
// if you comment the following two lines, the testcase will work, and the ClassLoader will be released.
// That proves that nothing is wrong with the test, and LocaleConvertUtilsBean is holding a reference
LocaleConvertUtils.register(new IntegerLocaleConverter(Locale.US, false), beanClass, Locale.US);
assertEquals(new Integer(12345), LocaleConvertUtils.convert(bean.toString(), beanClass, Locale.US, "#,###"));
// this should make the reference go away.
loader = null;
beanClass = null;
bean = null;
forceGarbageCollection(); /* Try to force the garbage collector to run by filling up memory */
if (someRef.get() != null) {
profilerLeakReport("LocaleConvertUtilsBean converters", className);
}
// if everything is fine, this will be null
assertNull("LocaleConvertUtilsBean is holding a reference to the classLoader", someRef.get());
// Clear All BeanUtils caches after the test
clearAllBeanUtilsCaches();
}
/**
* Clear all the BeanUtils Caches manually.
*
* This is probably overkill, but since we're dealing with static caches
* it seems sensible to ensure that all test cases start with a clean sheet.
*/
private void clearAllBeanUtilsCaches() {
// Clear BeanUtilsBean's PropertyUtilsBean descriptor caches
BeanUtilsBean.getInstance().getPropertyUtils().clearDescriptors();
// Clear LocaleBeanUtilsBean's PropertyUtilsBean descriptor caches
LocaleBeanUtilsBean.getInstance().getPropertyUtils().clearDescriptors();
// Clear MethodUtils's method cache
MethodUtils.clearCache();
// Clear WrapDynaClass cache
WrapDynaClass.clear();
// replace the existing BeanUtilsBean instance for the current class loader with a new, clean instance
BeanUtilsBean.setInstance(new BeanUtilsBean());
// replace the existing LocaleBeanUtilsBean instance for the current class loader with a new, clean instance
LocaleBeanUtilsBean.setInstance(new LocaleBeanUtilsBean());
}
/**
* Try to force the garbage collector to run by filling up memory and calling System.gc().
*/
private void forceGarbageCollection() throws Exception {
// Fill up memory
SoftReference ref = new SoftReference(new Object());
int count = 0;
while(ref.get() != null && count++ < 5) {
java.util.ArrayList list = new java.util.ArrayList();
try {
long i = 0;
while (true && ref.get() != null) {
list.add("A Big String A Big String A Big String A Big String A Big String A Big String A Big String A Big String A Big String A Big String " + (i++));
}
} catch (Throwable ignored) {
}
list.clear();
list = null;
// System.out.println("Count " + count + " : " + getMemoryStats());
System.gc();
Thread.sleep(1000);
}
// System.out.println("After GC: " + getMemoryStats());
if (ref.get() != null) {
throw new IllegalStateException("Your JVM is not releasing SoftReference, try running the testcase with less memory (-Xmx)");
}
}
/**
* Create a new class loader instance.
*/
private static URLClassLoader newClassLoader() throws MalformedURLException {
String dataFilePath = MemoryLeakTestCase.class.getResource("pojotests").getFile();
//System.out.println("dataFilePath: " + dataFilePath);
String location = "file://" + dataFilePath.substring(0,dataFilePath.length()-"org.apache.commons.beanutils.memoryleaktests.pojotests".length());
//System.out.println("location: " + location);
StringBuffer newString = new StringBuffer();
for (int i=0;i<location.length();i++) {
if (location.charAt(i)=='\\') {
newString.append("/");
} else {
newString.append(location.charAt(i));
}
}
String classLocation = newString.toString();
//System.out.println("classlocation: " + classLocation);
URLClassLoader theLoader = URLClassLoader.newInstance(new URL[]{new URL(classLocation)},null);
return theLoader;
}
/**
* Produce a profiler report about where the leaks are.
*
* This requires JBoss's profiler be installed, see:
* http://labs.jboss.com/jbossprofiler/
*
* @param className The name of the class to profile
*/
private void profilerLeakReport(String test, String className) {
/*
* If you want a report about where the leaks are... uncomment this,
* add jboss-profiler.jvmti.jar and jboss-commons.jar (for org.jboss.loggin).
* You will then have a report for where the references are.
System.out.println(" ----------------" + test + " START ----------------");
org.jboss.profiler.jvmti.JVMTIInterface jvmti = new org.jboss.profiler.jvmti.JVMTIInterface();
System.out.println(jvmti.exploreClassReferences(className, 8, true, true, true, false, false));
System.out.println(" ----------------" + test + " END ------------------");
*/
}
/**
* Test for JDK 1.5
*/
private boolean isPre15JVM() {
String version = System.getProperty("java.specification.version");
StringTokenizer tokenizer = new StringTokenizer(version,".");
if (tokenizer.nextToken().equals("1")) {
String minorVersion = tokenizer.nextToken();
if (minorVersion.equals("0")) return true;
if (minorVersion.equals("1")) return true;
if (minorVersion.equals("2")) return true;
if (minorVersion.equals("3")) return true;
if (minorVersion.equals("4")) return true;
}
return false;
}
/**
* Get the total, free, used memory stats.
* @return the total, free, used memory stats
*/
private String getMemoryStats() {
java.text.DecimalFormat fmt = new java.text.DecimalFormat("#,##0");
Runtime runtime = Runtime.getRuntime();
long free = runtime.freeMemory() / 1024;
long total = runtime.totalMemory() / 1024;
long used = total - free;
return "MEMORY - Total: " + fmt.format(total) + "k " + "Used: "
+ fmt.format(used) + "k " + "Free: "
+ fmt.format(free) + "k";
}
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?