Model is introduced

Proxy Design Pattern is a structural Design Pattern used to extend (enhance) the functionality of an existing class without changing its code.



The proxy pattern provides a proxy object to an object, which holds references to the original object and controls access to the original object. The object composition proxy object can completely override the capabilities of the original object, and can be added as needed to enhance the functionality and make the code more OCP compliant.



For the agent mode, Java developers are certainly not unfamiliar, Spring, Mybatis, Dubbo and other frameworks have a large number of use, master the agent mode of design and use, for us to read these excellent source code is of great benefit.



The following figure shows the UML class diagram of Proxy mode. Several subjects in the class diagram are client-client, Subject interface -Subject, RealSubject implementation class -RealSubject, and Proxy class -Proxy.



As can be seen from the class diagram, both RealSubject and Proxy implement the Subject interface, and Proxy holds the reference of Subject (actually RealSubject instance). The Client holds an instance of the Subject, but that instance is determined by the runtime (the Subject held by the Client is implemented as a Proxy instance).



The Proxy class is an intermediary that acts as an “intermediary” by handing over work that would have been done directly by a RealSubject. This middle tier does not just add code complexity, it can do a lot of work and benefits, as illustrated by a few business scenarios:

  • If you need to measure the time and error rate of the RealSubject# Request method, you can embed data in the Proxy without modifying the RealSubjce code. Statistical performance index is not the function of RealSubject, strong behavior to increase the logic will destroy its integrity, does not comply with OCP, SRP and other design principles; And if the RealSubject is a service provided by a third party, we cannot change its code. The use of Proxy Proxy class, better to achieve the separation of responsibilities, the original code without invasion, he good I good.
  • If you need to add Redis caching to the RealSubject#request method to improve interface performance, you can use the Proxy class to intercept user requests and add Redis caching logic. In Java, of course, we typically use Spring AOP for dynamic implementation, but the principle is the same.

In practice, there are static proxy and dynamic proxy. Static proxies are proxy classes that are implemented before code is compiled, either manually by programmers or automatically generated by tools; Dynamic proxies generate proxy classes at run time using reflection techniques. The above text description is rather dry, but the following example code illustrates the use of the two proxy modes. Continuing with the previous example, we need to add performance statistics logic to our existing business code, which looks like this:

public interface Subject {
    /** * interface method */
    void request(a);
}

public class RealSubject implements Subject {
    @Override
    public void request(a) {
        try {
            // Use sleep to simulate business execution
            Thread.sleep(50);
        } catch (Exception ex) {

        }
        System.out.println("RealSubject#request method invoked."); }}public class Client {
    public static void main(String[] args) {
        // instantiate the object
        Subject realSubject=new RealSubject();
        // method callrealSubject.request(); }}Copy the code

Static agent

Objective: to add performance statistics to RealSubject#request. For simplicity we’ll just do a time-consuming demo. According to the design idea of the proxy mode, we need to manually create a proxy class in the static proxy. The proxy class should also implement the Subject interface and contain a reference to the RealSubject. We need to add the invocation time calculation logic in the proxy class to complete the time statistics. Based on the above ideas, create the MetricProxy class, the code is as follows:

public class MetricProxy implements Subject {

    private Subject delegate;

    public MetricProxy(Subject delegate) {
        this.delegate = delegate;
    }

    @Override
    public void request(a) {
        // Method execution start time
        long start = (new Date()).getTime();
        // Invoke the request method of the delegate object
        this.delegate.request();
        // Method execution end time
        long stop = (new Date()).getTime();
        // Calculation time
        long span = stop - start;
        System.out.printf("invoke request() cost %d ms", span); }}Copy the code

Pass in the delegate object via the MetricProxy constructor and call the delegate#request method within the request method, ensuring that the original functionality remains unchanged; To increase time statistics, time logging and calculation logic are added before and after calling delegate#request. Next, we can call this in the Client:

public class Client {
    public static void main(String[] args) {
        // Create a delegate object
        Subject delegate = new RealSubject();
        // Create a proxy object
        Subject proxy = new MetricProxy(delegate);
        // Invoke the proxy object request methodproxy.request(); }}Copy the code

Taken as a whole, the static proxy approach is a “high-fidelity” implementation of the UML class diagram from the “Schema Introduction” section, and the two combined should give you an intuitive understanding of the proxy pattern. Here’s a question to consider: ** Now there is one Subject interface that needs to add performance statistics. What if there are 100 subjects like this interface that need to extend statistics? Write 100 proxy classes? Don’t be too sour! ** here, you need to dynamic proxy play.

A dynamic proxy

As mentioned earlier, dynamic proxies are used to dynamically implement proxy classes at run time using reflection techniques, without the developer being aware of the process. The Java SDK Reflect package provides Proxy, InvocationHandler and other utility classes for implementing dynamic proxies. Implementing dynamic proxies using Java is not a challenge. According to JDK requirements to create a proxy assistant class DynamicProxyHandler, the implementation of InvocationHandler interface, in its invoke method to achieve performance indicator statistics function.

public class DynamicProxyHandler implements InvocationHandler {
    // Delegate object
    private Object delegate;

    public DynamicProxyHandler(Object delegate) {
        this.delegate = delegate;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // Start execution time
        long start = (new Date()).getTime();

        // Call the delegate object's method
        Object result = method.invoke(delegate, args);

        // Call completion time
        long stop = (new Date()).getTime();
        // Calculate the duration
        long span = stop - start;
        System.out.printf("invoke request() cost %d ms", span);
        returnresult; }}Copy the code

Create a dynamic proxy object using Proxy#newProxyInstance:

public class Client {
    public static void main(String[] args) {
        // Add this sentence to generate a $proxy0.class file, which is the dynamically generated proxy class file
        //System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
        // Create a proxy helper class object
        DynamicProxyHandler proxy = new DynamicProxyHandler(new RealSubject());
        // Dynamically create a proxy class object that implements the Subject interface
        Subject subject = (Subject) Proxy.newProxyInstance(Subject.class.getClassLoader(), new Class[]{Subject.class}, proxy);
        // Call the methodsubject.request(); }}Copy the code

Run it and you’ll see exactly the same output as the static proxy example. At this point, we’ll feel no different from static proxies, and we’ll have implemented a proxy class with no less work. However, if you look closely, you’ll see that the DynamicProxyHandler is actually type independent. Here we pass in Subject, which can also pass other types, which is what makes it so powerful. Ever wonder what dynamically generated proxy classes in the JDK actually look like? How did the subject.request() code reach DynamicProxyHandler#invoke? Looking at line 4 of the Client#main method above, open the previous comment and run the code again to generate a $proxy0.class file under the package com.sun.proxy, which I pasted below:

public final class $Proxy0 extends Proxy implements Subject {
    private static Method m1;
    private static Method m2;
    private static Method m3;
    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 newUndeclaredThrowableException(var4); }}public final String toString(a) throws  {
        try {
            return (String)super.h.invoke(this, m2, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw newUndeclaredThrowableException(var3); }}public final void request(a) throws  {
        try {
            super.h.invoke(this, m3, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw newUndeclaredThrowableException(var3); }}public final int hashCode(a) throws  {
        try {
            return (Integer)super.h.invoke(this, m0, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw newUndeclaredThrowableException(var3); }}// reflection creates method objects
    static {
        try {
            m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
            m2 = Class.forName("java.lang.Object").getMethod("toString");
            m3 = Class.forName("com.raysonxin.proxy.staticproxy.Subject").getMethod("request");
            m0 = Class.forName("java.lang.Object").getMethod("hashCode");
        } catch (NoSuchMethodException var2) {
            throw new NoSuchMethodError(var2.getMessage());
        } catch (ClassNotFoundException var3) {
            throw newNoClassDefFoundError(var3.getMessage()); }}}Copy the code

$Proxy0 inherits the Proxy class and implements the Subject interface, so it has the ability to do both. Take the request method as an example: when the request method is called, it calls super.h.invoke(this, m3, (Object[])null); Here, super is the base Proxy and H is our DynamicProxyHandler, so DynamicProxyHandler#invoke will be called. The dynamic proxy implementation in Client is the basic implementation of JDK, and the actual development is often with the help of Spring AOP powerful section-oriented programming, based on Spring Bean, annotation and other technologies for dynamic implementation. In my last article “Mybatis source SQL execution process”, the creation of Mapper instance is a living example.

conclusion

The proxy pattern is a common, often used, often said classic design pattern in our development, to improve the extensibility of the program, make the code maintainable. Spring, Mybatis and other excellent Java development frameworks used in daily work are based on reflection + dynamic proxy, providing powerful functions and providing many conveniences for our development. This paper briefly introduces the design idea of proxy mode, and illustrates its classical static proxy and dynamic proxy with examples. Understand the implementation process by looking at the JDK generated proxy classes. Seeing is better than hearing, and practicing is better than seeing. I hope this article will give you a preliminary understanding of the agency model. My ability is limited, if you have any questions, please feel free to comment!