All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.alibaba.jvm.sandbox.agent.SandboxClassLoader Maven / Gradle / Ivy

There is a newer version: 1.4.0
Show newest version
package com.alibaba.jvm.sandbox.agent;

import java.io.Closeable;
import java.io.IOException;
import java.lang.reflect.Field;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Collection;
import java.util.Enumeration;
import java.util.jar.JarFile;

/**
 * 加载Sandbox用的ClassLoader
 * Created by [email protected] on 2016/10/26.
 */
class SandboxClassLoader extends URLClassLoader {

    private final String toString;
    private final String path;

    SandboxClassLoader(final String namespace,
                       final String sandboxCoreJarFilePath) throws MalformedURLException {
        super(new URL[]{new URL("file:" + sandboxCoreJarFilePath)});
        this.path = sandboxCoreJarFilePath;
        this.toString = String.format("SandboxClassLoader[namespace=%s;path=%s;]", namespace, path);
    }

    @Override
    public URL getResource(String name) {
        URL url = findResource(name);
        if (null != url) {
            return url;
        }
        url = super.getResource(name);
        return url;
    }

    @Override
    public Enumeration getResources(String name) throws IOException {
        Enumeration urls = findResources(name);
        if (null != urls) {
            return urls;
        }
        urls = super.getResources(name);
        return urls;
    }

    @Override
    protected synchronized Class loadClass(String name, boolean resolve) throws ClassNotFoundException {
        final Class loadedClass = findLoadedClass(name);
        if (loadedClass != null) {
            return loadedClass;
        }

//        // 优先从parent(SystemClassLoader)里加载系统类,避免抛出ClassNotFoundException
//        if(name != null && (name.startsWith("sun.") || name.startsWith("java."))) {
//            return super.loadClass(name, resolve);
//        }

        try {
            Class aClass = findClass(name);
            if (resolve) {
                resolveClass(aClass);
            }
            return aClass;
        } catch (Exception e) {
            return super.loadClass(name, resolve);
        }
    }

    @Override
    public String toString() {
        return toString;
    }


    /**
     * 尽可能关闭ClassLoader
     * 

* URLClassLoader会打开指定的URL资源,在SANDBOX中则是对应的Jar文件,如果不在shutdown的时候关闭ClassLoader,会导致下次再次加载 * 的时候,依然会访问到上次所打开的文件(底层被缓存起来了) *

* 在JDK1.7版本中,URLClassLoader提供了{@code close()}方法来完成这件事;但在JDK1.6版本就要下点手段了; *

* 该方法将会被{@code ControlModule#shutdown}通过反射调用, * 请保持方法声明一致 */ @SuppressWarnings("unused") public void closeIfPossible() { // 如果是JDK7+的版本, URLClassLoader实现了Closeable接口,直接调用即可 if (this instanceof Closeable) { try { ((Closeable) this).close(); } catch (Throwable cause) { // ignore... } return; } // 对于JDK6的版本,URLClassLoader要关闭起来就显得有点麻烦,这里弄了一大段代码来稍微处理下 // 而且还不能保证一定释放干净了,至少释放JAR文件句柄是没有什么问题了 try { final Object sun_misc_URLClassPath = forceGetDeclaredFieldValue(URLClassLoader.class, "ucp", this); final Object java_util_Collection = forceGetDeclaredFieldValue(sun_misc_URLClassPath.getClass(), "loaders", sun_misc_URLClassPath); for (final Object sun_misc_URLClassPath_JarLoader : ((Collection) java_util_Collection).toArray()) { try { final JarFile java_util_jar_JarFile = forceGetDeclaredFieldValue(sun_misc_URLClassPath_JarLoader.getClass(), "jar", sun_misc_URLClassPath_JarLoader); java_util_jar_JarFile.close(); } catch (Throwable t) { // if we got this far, this is probably not a JAR loader so skip it } } } catch (Throwable cause) { // ignore... } } private T forceGetDeclaredFieldValue(Class clazz, String name, Object target) throws NoSuchFieldException, IllegalAccessException { final Field field = clazz.getDeclaredField(name); field.setAccessible(true); return (T) field.get(target); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy