美文网首页
反射代理类加载器的潜在内存使用问题

反射代理类加载器的潜在内存使用问题

作者: 行径行 | 来源:发表于2017-11-19 10:54 被阅读0次

  • tags:反射
  • categories: problems
  • date: 2017-05-28 14:50:04

使用反射代理类加载器的潜在内存使用问题

大量的类加载器 “sun/reflect/DelegatingClassLoader”,用来加载“sun/reflect/GeneratedMethodAccessor”类,可能导致潜在的占用大量本机内存空间问题,应用服务器进程占用的内存会显著增大。您还有可能遇到抛出的内存溢出错误。案例:假笨说-从一起GC血案谈到反射原理

先把结论说明了:

  • 在上述案例中,得到的结论是:反射类加载器导致Perm溢出。系统使用jdk1.7,且GC收集器是G1。这个版本的G1的特性是只有在Full GC的情况下才会对perm(永久区)里的类进行卸载,正常的G1的gc过程是不会对Perm中的类进行卸载的,所以,当Perm内存被类堆积满的时候,就会进行一次Full Gc将无用的类卸载掉。那么为什么会产生那么多类在Perm区域中呢,通过案例中GC日志的分析可以知道,是因为产生了大量的"sun.reflect.DelegatingClassLoader",那么为什么会有那么多的代理委托类加载器,用于加载什么类的呢?
    从分析可以知道,是因为使用三方的Xfire协议,该协议过程中产生大量的RPC,将得到结果进行反序列化时候,是通过Method.invoke反射原理来实现目的的,在Xfire实现中,内部还有Methodref等包含软索引SoftReference的引用,很容易就会被G1给回收了,一旦回收了,程序内部就会Copy来创建一个新的Method,在调用其invoke通过MethodAccessor来实际调用,多次操作大于指定阈值(15)就会创建一个“sun/reflect/GeneratedMethodAccessor”类字节码,然后在通过DelegatingClassLoader来加载该字节码,就会将这些类都保存到Perm中,如此复返,就会Perm溢出。

  • 当使用Java反射时,Java虚拟机有两种方法获取被反射的类的信息。它可以使用一个JNI存取器。如果使用Java字节码存取器,则需要拥有它自己的Java类和类加载器(sun/reflect/GeneratedMethodAccessor类和sun/reflect/DelegatingClassLoader)。这些类和类加载器使用本机内存。字节码存取器也可以被JIT编译,这样会增加本机内存的使用。如果Java反射被频繁使用,会显著地增加本机内存的使用。
    Java虚拟机会首先使用JNI存取器,然后在访问了同一个类若干次后,会改为使用Java字节码存取器。这种当Java虚拟机从JNI存取器改为字节码存取器的行为被称为膨胀。幸运的是,我们可以通过一个Java属性控制这种行为。属性sun.reflect.inflationThreshold会告诉Java虚拟机使用JNI存取器多少次。如果设为0,则总是使用JNI存取器。由于字节码存取器比JNI存取器使用更多本机内存,当我们看到大量Java反射时,最好使用JNI存取器。我们只需要设置inflationThreshold属性值为0即可。

    image

Reflection反射原理

下面通过一个简单的反射例子来捋一捋java内部反射机制过程。

public class ReflectDemo {

    
    public static void main(String[] args) throws Exception{
        Proxy target = new ReflectDemo.Proxy();
        Method method = Proxy.class.getDeclaredMethod("pmethod", null);
        //MethodAccessor.invoke
        method.invoke(target, null);
        
    }
    
    static class Proxy{
        public void pmethod(){
            System.out.println("Proxy.pmethod");
        }
    }
}

上述可以看到通过Class.getDeclaredMethod反射方法获取Proxy类中pmethod方法,最后成功调用。先从下图中看看从方法调用到最后方法执行的流程图:

image

获取得到Method对象

根据上面的例子,当通过Class.getDeclaredMethod(MethodName)进行反射,获取指定类的指定的方法的时候,具体是如何进行的?结合上面的流程图,分步说明:

// Class.java
  @CallerSensitive
    public Method getDeclaredMethod(String name, Class<?>... parameterTypes)
        throws NoSuchMethodException, SecurityException {
        // be very careful not to change the stack depth of this
        // checkMemberAccess call for security reasons
        // see java.lang.SecurityManager.checkMemberAccess
        checkMemberAccess(Member.DECLARED, Reflection.getCallerClass(), true);
        Method method = searchMethods(privateGetDeclaredMethods(false), name, parameterTypes);
        if (method == null) {
            throw new NoSuchMethodException(getName() + "." + name + argumentTypesToString(parameterTypes));
        }
        return method;
    }

可以看到,先调用privateGetDeclaredMethods方法,在调用searchMethods方法最后得到一个Method对象。
其中,在看到privateGetDeclaredMethods方法之前,需要知道Class类中有个很重要的属性:ReflectionData,这个属性类中就是保存着每次从JVM中获取指定类时候类中的属性,比如方法,属性字段等:

static class ReflectionData<T>{
  volatile Field[] declaredFields;
  volatile Field[] publicFields;
  volatile Method[] declaredMethods;
  volatile Method[] publicMethods;
  volatile Constructor<T>[] declaredConstructors;
  volatile Constructor<T>[] publicConstructors;
  //Intermediate results for getFields and getMethods
  volatile Field[] declaredPublicFields;
  volatile Method[] declaredPublicMethods;
  //value of classRedefineCount when we create this reflectionData instance
  final int redefinedCount;
  
  ReflectionData(int redefinedCount){
    this.redefinedCount = redefinedCount;
  }

}

这个属性是软引用SoftReference的,也就是在某些内存比较紧张的情况下,是会被GC回收的,可以通过JVM参数:-XX:SoftRefLRUPolicyMSPerMB来控制回收时机。所以,一旦某个类的ReflectionData属性被回收了的话,意味着当第二次再想通过这个属性来获取反射信息的时候,就会发现,缓存中没有了,就会重新创建一个新的ReflectionData对象,将从JVM中获取到类的信息封装到属性中,那么这个类的属性类中的关联所有Method,Field等属性对象都是重新创建的。重新创建对象必然会消耗内存资源和一些时间,有一些副作用也是肯定的。

然后,现在新的JDK版本,将ReflactionData对象给取代了,取代方式是在Class.java类中通过声明反射的软引用集合属性:

//java.lang.Class
    /**
     * Reflection support.
     */

    // Caches for certain reflective results
    private static boolean useCaches = true;
    private volatile transient SoftReference<Field[]> declaredFields;
    private volatile transient SoftReference<Field[]> publicFields;
    private volatile transient SoftReference<Method[]> declaredMethods;
    private volatile transient SoftReference<Method[]> publicMethods;
    private volatile transient SoftReference<Constructor<T>[]> declaredConstructors;
    private volatile transient SoftReference<Constructor<T>[]> publicConstructors;
    // Intermediate results for getFields and getMethods
    private volatile transient SoftReference<Field[]> declaredPublicFields;
    private volatile transient SoftReference<Method[]> declaredPublicMethods;

    // Incremented by the VM on each call to JVM TI RedefineClasses()
    // that redefines this class or a superclass.
    private volatile transient int classRedefinedCount = 0;

    // Value of classRedefinedCount when we last cleared the cached values
    // that are sensitive to class redefinition.
    private volatile transient int lastRedefinedCount = 0;

(1) privateGetDeclaredMethods方法:
从缓存或者JVM中获取该Class中,符合反射方法调用传递过出来方法名称,方法参数类型的Method对象列表。

    private Method[] privateGetDeclaredMethods(boolean publicOnly) {
        checkInitted();
        Method[] res = null;
        if (useCaches) {
            clearCachesOnClassRedefinition();
            if (publicOnly) {
                if (declaredPublicMethods != null) {
                    res = declaredPublicMethods.get();
                }
            } else {
                if (declaredMethods != null) {
                    res = declaredMethods.get();
                }
            }
            if (res != null) return res;
        }
        //若是在当前反射缓冲中,所有的Method,Field软引用数组中都没有找到,
        //则调用Reflection.filterMethods方法从JVM内部获取,将得到的数据
        //封装到Class的反射引用字段数组中,缓存起来
        // No cached value available; request value from VM
        res = Reflection.filterMethods(this, getDeclaredMethods0(publicOnly));
        if (useCaches) {
            if (publicOnly) {
                
                declaredPublicMethods = new SoftReference<>(res);
            } else {
                //重新缓存
                declaredMethods = new SoftReference<>(res);
            }
        }
        return res;
    }

(2) searchMethods方法调用:
searchMethods将从privateGetDeclaredMethods返回的方法列表里找到一个同名的匹配的方法,然后复制一个新的Method方法对象出来。

    private static Method searchMethods(Method[] methods,
                                        String name,
                                        Class<?>[] parameterTypes)
    {
        Method res = null;
        String internedName = name.intern();
        for (int i = 0; i < methods.length; i++) {
            Method m = methods[i];
            if (m.getName() == internedName
                && arrayContentsEq(parameterTypes, m.getParameterTypes())
                && (res == null
                    || res.getReturnType().isAssignableFrom(m.getReturnType())))
                res = m;
        }
        //拷贝一个Method对象
        return (res == null ? res : getReflectionFactory().copyMethod(res));
    }

(3) ReflectionFactory.copyMethod()方法:
当在第二步中,在方法列表中匹配到同名同参数的Method对象的时候,这时候就调用ReflectionFactory.copyMehthod方法,拷贝一个一样方法。在ReflectionFactory中有个LangReflectAccess字段,用于解决反射访问其他包中,私有的,共有的方法或者字段属性的权限问题。

//ReflectionFactory.java
// Provides access to package-private mechanisms in java.lang.reflect
private static volatile LangReflectAccess langReflectAccess;

  
/** Makes a copy of the passed method. The returned method is a
        "child" of the passed one; see the comments in Method.java for
        details. */
public Method copyMethod(Method arg) {
        return langReflectAccess().copyMethod(arg);
 }



(4) ReflectAccess.copyMethod方法:
上述第三步骤中ReflectionFactory.copyMethod中的langReflectionAccess()方法就是返回一个LangReflectAccess接口类,而ReflectAccess则是该接口的实现类,所以就会调用该类的copyMethod()方法:

   //
    // Copying routines, needed to quickly fabricate new Field,
    // Method, and Constructor objects from templates
    //
    public Method      copyMethod(Method arg) {
        return arg.copy();
    }

可以,最后实质是调用传入的参数Method对象的copy方法。
(5) Method.copy()方法:
Method对象的copy方法,主要是通过将传入的Method对象的方法名,参数,等等其他属性都拷贝一份,但是Method对象的methodAccessor字段却是共享的。通过Method类中root字段属性来实现。

//Method.java
    // For sharing of MethodAccessors. This branching structure is
    // currently only two levels deep (i.e., one root Method and
    // potentially many Method objects pointing to it.)
    private Method              root;

    //共享的methodAccessor
    private volatile MethodAccessor methodAccessor;


    Method copy() {
        // This routine enables sharing of MethodAccessor objects
        // among Method objects which refer to the same underlying
        // method in the VM. (All of this contortion is only necessary
        // because of the "accessibility" bit in AccessibleObject,
        // which implicitly requires that new java.lang.reflect
        // objects be fabricated for each reflective call on Class
        // objects.)
        Method res = new Method(clazz, name, parameterTypes, returnType,
                                exceptionTypes, modifiers, slot, signature,
                                annotations, parameterAnnotations, annotationDefault);
        res.root = this;
        // Might as well eagerly propagate this if already present
        res.methodAccessor = methodAccessor;
        return res;//返回拷贝的Method对象
    }

调用Method.invoke方法

通过上面的步骤,就能将目标类的指定反射方法的拷贝对象给获取到了。就如例子中Method method = Proxy.class.getDeclaredMethod("pmethod", null)这句执行完成了。下面就是调用方法的过程了。

整个Method.invoke方法内部实际调用是MethodAccessor接口实现类的invoke方法调用。Method.invoke只是表面的壳而已,也可以说是代理调用。下面就说说MethodAccessor接口实现类和实际调用过程。

MethodAccessor接口的声明:

public interface MethodAccessor {
    /** Matches specification in {@link java.lang.reflect.Method} */
    public Object invoke(Object obj, Object[] args)
        throws IllegalArgumentException, InvocationTargetException;
}

该接口的实现类有以下几种:

  • NativeMethodAccessorImpl
  • DelegatingMethodAccessorImpl
  • GeneratedMethodAccessorXXX

其中的第一个NativeMethodAccessorImpl对应的就是通过JNI(java本地接口)存取器来获取反射类字节码信息。第三个GeneratedMethodAccessor<Num> 类就是通过代理类加载器DelegatingClassLoader来加载反射类字节码的。中间的代理MethodAccessorImpl则是可以将传入的MethodAccessor参数对象,注入给Method对象的methodAccessor属性,实现代理调用:也就是说明,通常情况下,我们调用某个Method对象的invoke方法,内部都是通过这个代理类来实现调用的。

class DelegatingMethodAccessorImpl extends MethodAccessorImpl {
    private MethodAccessorImpl delegate; //代理模式,实际调用的接口实现

    DelegatingMethodAccessorImpl(MethodAccessorImpl delegate) {
        setDelegate(delegate);
    }

    public Object invoke(Object obj, Object[] args)
        throws IllegalArgumentException, InvocationTargetException
    {
        return delegate.invoke(obj, args);//代理调用
    }

    void setDelegate(MethodAccessorImpl delegate) {
        this.delegate = delegate;
    }
}

(1) Method.invoke方法调用:
我们代码显示调用的Method对象的invoke对象,虽然已经知道实际上是MethodAccessor实现类底层调用,但是也会是可以看看该方法内部源代码:

   @CallerSensitive
    public Object invoke(Object obj, Object... args)
        throws IllegalAccessException, IllegalArgumentException,
           InvocationTargetException
    {
        if (!override) {
            if (!Reflection.quickCheckMemberAccess(clazz, modifiers)) {
                // Until there is hotspot @CallerSensitive support
                // can't call Reflection.getCallerClass() here
                // Workaround for now: add a frame getCallerClass to
                // make the caller at stack depth 2
                Class<?> caller = getCallerClass();
                checkAccess(caller, clazz, obj, modifiers);
            }
        }
        //Method类共享methodAccessor对象,判断是否为null,若是null,则调用指定方法获取
        MethodAccessor ma = methodAccessor;             // read volatile
        if (ma == null) {
            //调用以下方法来获取MethodAccessor实现类
            ma = acquireMethodAccessor();
        }
        return ma.invoke(obj, args);
    }

(2) acquireMethodAccessor方法获取MethodAccessor:
当某个反射方法的methodAccessor属性字段为null的时候,需要显示的调用acquireMethodAccessor方法来获取方法访问实现对象:

    private MethodAccessor acquireMethodAccessor() {
        // First check to see if one has been created yet, and take it
        // if so
        MethodAccessor tmp = null;
        if (root != null) tmp = root.getMethodAccessor();
        if (tmp != null) {
            methodAccessor = tmp;
        } else {
            // Otherwise fabricate one and propagate it up to the root
            //从反射工厂类中,显示创建一个MethodAccessor实现对象
            tmp = reflectionFactory.newMethodAccessor(this);
            setMethodAccessor(tmp);
        }

        return tmp;
    }

因为在通过反射获取Method对象,在copy的时候,有个语句是res.root = this,因为MethodAccessor是共享的,所以,可以先从父方法访问器中获取父类Method对象的methodAccessor属性对象,判读是否为null,不为null的话,直接返回。若是methodAccessor还是为空,则是需要显示的调用ReflectionFactory.newMethodAccessor()方法。

(3)ReflectionFactory.newMethodAccessor()调用:
通过newMethodAccessor()方法,创建一个新的MethodAccessor对象。ReflectionFactory类中有几个与MethodAccessor相关的属性字段:noInflation,inflationThreshold。可以先看看创建新方法访问对象的源代码:

    public MethodAccessor newMethodAccessor(Method method) {
        checkInitted();
        //若是noInflation设置为true,则直接调用MethodAccessorGenerator创建新对象访问对象
        if (noInflation) {
            return new MethodAccessorGenerator().
                generateMethod(method.getDeclaringClass(),
                               method.getName(),
                               method.getParameterTypes(),
                               method.getReturnType(),
                               method.getExceptionTypes(),
                               method.getModifiers());
        } else {
            NativeMethodAccessorImpl acc =
                new NativeMethodAccessorImpl(method);
            DelegatingMethodAccessorImpl res =
                new DelegatingMethodAccessorImpl(acc);
            acc.setParent(res);
            return res;
        }
    }

从源代码中可以看到有两个分支:一个当设置noInflation属性为true的时候,会直接调用MethodAccessorGenerator类直接创建新的MethodAccessor,内部的generatedMethod方法实现在稍后说明;另外一个分支中,这个分支也是常用的分支,因为noInflation默认设置为false。就来看看这个分支中是如何创建新的MethodAccessor对象的。

从源代码中可以看到,主要的实现就是通过JNI存取器调用本地代码库NativeMethodAccessorImpl来创建对象访问对象,当创建好对象后就通过DelegatingMethodAccessorImpl对象进行注入代理。

(4) NativeMethodAccessorImpl.invoke方法:
NativeMethodAccessorImpl类中会有一个字段numInvocations,用于计算使用JNI本地代码来生成MethodAccessor对象的次数,当调用次数大于15次的时候,就会调用MethodAccessorGenerator.generateMethod生成类名字形如GeneratedMethodAccessorXXX的字节码类文件。该类字节码是通过DelegatingClassLoader来加载的。
为什么要设计这种机制呢?
java的反射机制运行效率是比较低的,执行Method.invoke()或Constructor.newInstance()都是通过调用native方法完成的,JDK为了提高反射运行效率,引入了一个机制叫“Inflation”,它首先通过DelegatingClassLoader去加载字节码,再执行相关的逻辑,字节码会缓存起来,所以第一次有加载的成本比正常执行慢3-4倍,但是后面的执行会有20倍以上的性能提升,这样整体性能会有很大的提升。当然,这种机制也会有弊端,放在后面说。

    public Object invoke(Object obj, Object[] args)
        throws IllegalArgumentException, InvocationTargetException
    {   //大于15次创建GeneratedMethodAccessorXXX字节码
        if (++numInvocations > ReflectionFactory.inflationThreshold()) {
            MethodAccessorImpl acc = (MethodAccessorImpl)
                new MethodAccessorGenerator().
                    generateMethod(method.getDeclaringClass(),
                                   method.getName(),
                                   method.getParameterTypes(),
                                   method.getReturnType(),
                                   method.getExceptionTypes(),
                                   method.getModifiers());
            parent.setDelegate(acc);
        }
        //默认调用JNI存取器来返回一个MethodAccessor对象
        return invoke0(method, obj, args);
    }

(5) MethodAccessorGenerator.generateMethod方法调用:
当noInflation设置为true,即不使用Inflation机制,那么当Method对象的methodAccessor属性为null的时候,就会直接调用MethodAccessorGenerator.generateMethod生成代理字节类GeneratedMethodAccessorXXX,并在内存中缓存起来;还要一种就是调用JNI的NativeMethodAccessorImpl.invoke次数大于15这个阈值的时候,也会调用这个方法生成代理字节类。

// MethodAccessorGenerator.java
  private MagicAccessorImpl generate(final Class declaringClass,
                                       String name,
                                       Class[] parameterTypes,
                                       Class   returnType,
                                       Class[] checkedExceptions,
                                       int modifiers,
                                       boolean isConstructor,
                                       boolean forSerialization,
                                       Class serializationTargetClass)
    {
        ByteVector vec = ByteVectorFactory.create();
        asm = new ClassFileAssembler(vec);
        this.declaringClass = declaringClass;
        this.parameterTypes = parameterTypes;
        this.returnType = returnType;
        this.modifiers = modifiers;
        this.isConstructor = isConstructor;
        this.forSerialization = forSerialization;

        //通过ClassFileAssembler类型的asm对象,设置Class字节码内容,
        //也就是通过代码来构建一个符合JVM规范的Class字节码文件
         asm.emitMagicAndVersion();//设置魔数和版本.....
        ....
        //这里有个重点,就是生成的字节码文件的名字规则
        //“GeneratedMethodAccessor+Num” NUm为调用次数
        final String generatedName = generateName(isConstructor, forSerialization);
        
        //通过ClassDefiner.defineClass方法,指定代理类加载器来加载这个类字节码文件
        //从而返回MethodAccessorImpl对象
        return AccessController.doPrivileged(
            new PrivilegedAction<MagicAccessorImpl>() {
                public MagicAccessorImpl run() {
                        try {
                        return (MagicAccessorImpl)
                        //bytes是构建好的class字节码文件
                        ClassDefiner.defineClass
                                (generatedName,
                                 bytes,
                                 0,
                                 bytes.length,
                                 declaringClass.getClassLoader()).newInstance();
                        } catch (InstantiationException e) {
                            throw (InternalError)
                                new InternalError().initCause(e);
                        } catch (IllegalAccessException e) {
                            throw (InternalError)
                                new InternalError().initCause(e);
                        }
                    }
                });
    }

    //生成的class类字节码文件的命名规则
    private static synchronized String generateName(boolean isConstructor,
                                                    boolean forSerialization)
    {
        if (isConstructor) {
            if (forSerialization) {
                int num = ++serializationConstructorSymnum;
                return "sun/reflect/GeneratedSerializationConstructorAccessor" + num;
            } else {
                int num = ++constructorSymnum;
                return "sun/reflect/GeneratedConstructorAccessor" + num;
            }
        } else {
            int num = ++methodSymnum;
            return "sun/reflect/GeneratedMethodAccessor" + num;
        }
    }

可以看到这个generate方法主要做了一下几件事情:

  • 通过ClassFileAssembler类创建了符合JVM规范的MethodAccessor接口实现类的字节码数组bytes。
  • 通过generateName方法,生成class类字节码文件:若是通过构造器反射调用,字节类文件名形如:"GeneratedConstructorAccessorXXXX",若是直接调用反射方法,不通过构造器,则如:"GeneratedMethodAccessorXXX"。
  • 通过ClassDefiner.defineClass创建DelegatingClassLoader代理类加载器,用于加载上面生成的class字节码,生成GeneratedMethodAccessorXXX类对象到内存中,以供Method.invoke方法调用。

当使用Method.invoke多次调用,生成GeneratedMethodAccessorXXX,并且使用DelegatingClassLoader加载该类的时候,通过DelegatedMethodAccessorImpl将GeneratedMethodAccessorXXX注入到目标方法的methodAccessor,所以实际也就是调用GeneratedMethodAccessorXXX.invoke()方法,通过上图代码可知,也就是调用目标对象的方法,和正常的方法调用一样。

(6) DelegatingClassLoader类加载器:
当通过MethodAccessorGenerator.generateMethod生成形如"GeneratedMethodAccessorXXX"的类字节对象,要加载到内存中使用,必须要通过类加载器来操作。那么,这些类字节文件就是被DelegatingClassLoader这个类加载器加载到内存中并进行装配使用的。

//ClassDefiner.class
    static Class defineClass(String name, byte[] bytes, int off, int len,
                             final ClassLoader parentClassLoader)
    {
        ClassLoader newLoader = AccessController.doPrivileged(
            new PrivilegedAction<ClassLoader>() {
                public ClassLoader run() {
                        return new DelegatingClassLoader(parentClassLoader);
                    }
                });
        return unsafe.defineClass(name, bytes, off, len, newLoader, null);
    }
}

加载到内存中后,就被缓存到java.lang.Class.class中的与反射有关的软引用SoftReference给缓存起来,如:private volatile transient SoftReference<Method[]> declaredMethods。那么这个Inflation机制有哪些弊端呢?
Inflation机制提高了反射的性能,但是对于重度使用反射的项目可能存在隐患,它带来了两个问题:
(1)初次加载的性能损失;
(2)动态加载的字节码导致PermGen持续增长;

当然了,解决方法也会有的,参照一下几篇文章:
假笨说-从一起GC血案谈到反射原理
使用反射代理类加载器的潜在内存使用问题
理解 JVM 如何使用 AIX 上的本机内存
sun.reflect.DelegatingClassLoader

相关文章

  • 反射代理类加载器的潜在内存使用问题

    tags:反射 categories: problems date: 2017-05-28 14:50:04 使用...

  • 玩转java反射

    原文 类加载器什么是类加载器类加载器的职责类加载器的组成 反射什么是反射如何使用反射反射在Android中的应用 ...

  • Java基础学习第二十七天——类加载器和反射

    [TOC] 第十三章 类加载器和反射 类的加载 当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载...

  • 2020-07-04【反射】

    类加载 类加载器 反射概述 获取Class类的对象 反射获取构造方法并使用 反射获取成员遍历并使用 反射获取成员方...

  • Java基础:反射

    反射注解动态代理相关阅读 Java基础:类加载器 Java基础:反射 Java基础:注解 Java基础:动态代理 ...

  • 《设计模式之禅》读书笔记-2.6-代理模式

    2.6 代理模式 代理的原理可先参看此文章:Java基础-001-枚举、反射、类加载器、内省、注解、泛型、代理 定...

  • java反射总结

    01_反射(类的加载概述和加载时机) A:类的加载概述当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通...

  • Java基础:类加载器

    系列阅读 Java基础:类加载器 Java基础:反射 Java基础:注解 Java基础:动态代理 1. 什么是类加...

  • JVM类加载和双亲委派机制

    类加载器 类加载器的作用就是把磁盘中的类文件加载到内存的方法区以供使用,分析类加载前,先看下jvm运行时都需要加载...

  • Java基础:注解

    系列阅读 Java基础:类加载器 Java基础:反射 Java基础:注解 Java基础:动态代理 1. 概述 注解...

网友评论

      本文标题:反射代理类加载器的潜在内存使用问题

      本文链接:https://www.haomeiwen.com/subject/druuvxtx.html