I wrote a Jdk dynamic proxy class to test the process of Jdk dynamic proxy class

  1. IRentingHouse rentingHouse = new RentingHouseImpl (a) ; // Client — client
public class RentingHouseImpl implements IRentingHouse { @Override public void rentHosue () { System.out.println ( "I'd like to rent a one-bedroom house"); } } public interface IRentingHouse { void rentHosue () ; }Copy the code
  1. Create dynamic proxy object methods in a dynamic proxy factory
Public Object getJdkProxy (Object obj) {public Object getJdkProxy (Object obj) Proxy.newProxyInstance( obj.getClass () .getClassLoader () , obj.getClass () .getInterfaces () , new InvocationHandler () { @Override public Object invoke ( Object proxy, Method method, Object [] args ) throws Throwable { Object result = null; // write the enhanced logic system.out.println (" agent (agent) charge 3000 yuan "); Result = method.invoke (obj,args); System.out.println (" Customer info sold for 30 cents "); return result; }}); }Copy the code
  1. Save classes produced by JDK dynamic proxies (see generated dynamic proxy objects)
@param filePath save path, */ private static void saveProxyFile (String... filePath ) { if ( filePath.length == 0 ) { System.getProperties() .put ( "sun.misc.ProxyGenerator.saveGeneratedFiles", "true" ) ; } else { FileOutputStream out = null; try { byte [] classFile = ProxyGenerator.generateProxyClass( "$Proxy0", RentingHouseImpl.class.getInterfaces ()) ; out = new FileOutputStream ( filePath [ 0 ] + "$Proxy0.class" ) ; out.write ( classFile ) ; } catch ( Exception e ) { e.printStackTrace () ; } finally { try { if ( out ! = null ) { out.flush () ; out.close () ; } } catch ( IOException e ) { e.printStackTrace () ; }}}}Copy the code
  1. The entrance
public static void main ( String [] args ) { saveProxyFile() ; IRentingHouse rentingHouse = new RentingHouseImpl () ; IRentingHouse jdkProxy = (IRentingHouse) proxyFactory.getInstance ().getjdkProxy () rentingHouse ) ; jdkProxy.rentHosue () ; }Copy the code

Source tracking

  1. Proxy.NewProxyInstance ()
  2. Class < ? > cl = getProxyClass0( loader, intfs ) ;

  1. proxyClassCache.get ( loader, interfaces ); ==>V value = supplier.get(a); ==>Objects.requireNonNull( valueFactory.apply ( key, parameter )) ;
  2. The code goes to the class ProxyClassFactory#apply() in java.lang.reflect.proxy.

Such a line. Line 639

byte [] proxyClassFile = ProxyGenerator.generateProxyClass( proxyName, interfaces, accessFlags ) ; Construct a byte proxy class. I passed it back later by decompilatingCopy the code

So we decompile byte [] proxyClassFile to get the class of the proxy object. Can directly call methods in saveProxyFile ProxyGenerator. GenerateProxyClass (), will be decompiled preserved in the class.

3. Proxy classes

// // Source code recreated from a .class file by IntelliJ IDEA // (powered by Fernflower decompiler) // package com.sun.proxy; import designpattern.proxy.dynamicproxy.IRentingHouse; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.lang.reflect.UndeclaredThrowableException; public final class $Proxy0 extends Proxy implements IRentingHouse { private static Method m1; private static Method m3; private static Method m2; private static Method m0; public $Proxy0(InvocationHandler var1) throws { super(var1); } public final boolean equals(Object var1) throws { try { return (Boolean)super.h.invoke(this, m1, new Object[]{var1}); } catch (RuntimeException | Error var3) { throw var3; } catch (Throwable var4) { throw new UndeclaredThrowableException(var4); } } public final void rentHosue() throws { try { super.h.invoke(this, m3, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } public final String toString() throws { try { return (String)super.h.invoke(this, m2, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } public final int hashCode() throws { try { return (Integer)super.h.invoke(this, m0, (Object[])null); } catch (RuntimeException | Error var2) { throw var2; } catch (Throwable var3) { throw new UndeclaredThrowableException(var3); } } static { try { m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object")); m3 = Class.forName("designpattern.proxy.dynamicproxy.IRentingHouse").getMethod("rentHosue"); m2 = Class.forName("java.lang.Object").getMethod("toString"); m0 = Class.forName("java.lang.Object").getMethod("hashCode"); } catch (NoSuchMethodException var2) { throw new NoSuchMethodError(var2.getMessage()); } catch (ClassNotFoundException var3) { throw new NoClassDefFoundError(var3.getMessage()); }}}Copy the code

Visible:

public final class $Proxy0 extends Proxy implements IRentingHouse {

Inherit Proxy, to achieve IRentingHouse

Conclusion Analysis:

Because the JDK dynamic Proxy classes have inherited the Proxy class – this class, so can establish contact with the Proxy class through the interface (two classes to build links, one is the inheritance relationship between the JDK already can’t pass this way, because the Java only supports single inheritance], another is to implement the same interface) the JDK dynamic Proxy to choose this 】 【, The proxied class must also be required to implement an interface through which the proxied class can communicate with the proxied class