Heim >Java >JavaBase >Was ist ein Java-Proxy?

Was ist ein Java-Proxy?

angryTom
angryTomOriginal
2019-11-13 11:37:374728Durchsuche

Was ist ein Java-Proxy?

Was ist ein Java-Proxy?

Was ist ein Proxy? Der Zugriff auf das Ziel erfolgt nicht direkt, sondern über eine Zwischenschicht, wie folgt:

Was ist ein Java-Proxy?

Statischer Java-Proxy

Wenn beispielsweise einige unserer Früchte, wie Bananen, Äpfel usw., in Java-Code geschrieben sind, sehen sie wahrscheinlich so aus:

(Empfohlenes Tutorial: Java-Tutorial )

//Fruit.java/**
 * 水果的接口
 */public interface Fruit {    /**
     * 获取水果的名字
     */
    public String getName();
}//Apple.javapublic class Apple implements Fruit {    @Override
    public String getName() {        return "苹果";
    }
}//Banana.javapublic class Banana implements Fruit {    @Override
    public String getName() {        return "香蕉";
    }
}

Um Obst zu essen, muss man es schälen. Man kann nicht für jede Frucht eine Unterklasse schreiben. Deshalb können wir ein Mittel herstellen und den Apfel schälen, bevor wir ihn essen. Wickeln Sie die ursprüngliche Frucht wie folgt in eine Schicht:


//PeelFruitProxy.java/**
 * 代理,让每个水果去皮
 */public class PeelFruitProxy implements Fruit {    private Fruit mFruit;    public PeelFruit(Fruit fruit) {        this.mFruit = fruit;
    }    @Override
    public String getName() {
        System.out.println("proxt:" + proxy.getClass().getName());        return "去皮的" + mFruit.getName();
    }
}

Fügen Sie eine Testklasse hinzu. Die Testklasse lautet wie folgt:

//Main.javapublic class Main {    public static void main(String[] args) {
        Apple apple=new Apple();//原始的苹果
        Banana banana=new Banana();//原始的香蕉

        PeelFruitProxy peelApple=new PeelFruitProxy(apple);//代理,添加去皮功能
        PeelFruitProxy peelBanana=new PeelFruitProxy(banana);//代理,添加去皮功能
        System.out.println(peelApple.getName());
        System.out.println(peelBanana.getName());
    }
}

Das Obige ist der statische Proxy von Java Vereinfacht ausgedrückt bedeutet das, das ursprüngliche Zielobjekt mit einer Ebene zu umhüllen, neue Dinge hinzuzufügen und dann das Ziel selbst aufzurufen. Aber wenn es sich nur um einen solchen statischen Proxy handelt, erfordert eine Schnittstelle einen Proxy, und ist die Implementierung sehr umständlich?

Dynamischer Proxy von Java

In Java gibt es eine Klasse namens Proxy, die dies erledigt, und Sie können Reflektion und Proxy-Abfangen direkt verwenden. Lassen Sie uns diese Klasse zunächst kurz vorstellen. Tatsächlich ist die am häufigsten verwendete statische Methode Proxt.newProxyInstance(), die so aussieht:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)

Zuerst müssen wir InvocationHandler implementieren, die Invoke-Methode implementieren und aufrufen Zum Zeitpunkt des Zielobjekts wird die Aufrufmethode zuerst aufgerufen, und der Implementierer muss die aufgerufene Methode in dieser Methode aktiv aufrufen.

//FruitInvocationHandler.java/**
 * 调用方法拦截器
 */public class FruitInvocationHandler implements InvocationHandler {    private Fruit mFruit;    public FruitInvocationHandler(Fruit fruit) {        this.mFruit = fruit;
    }    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        String result = (String) method.invoke(mFruit, args);//需要在这个方法里面,主动调用被代理的对象。
        return "去皮的" + result;
    }
}

Führen Sie es aus:

//Main.Javapublic class Main {    public static void main(String[] args) {
        Apple apple = new Apple();
        Fruit proxyApple = (Fruit) Proxy.newProxyInstance(Fruit.class.getClassLoader(), new Class[]{Fruit.class}, new FruitInvocationHandler(apple));
        System.out.println(proxyApple.getClass().getName());
        System.out.println(proxyApple.getName());

        Banana banana = new Banana();
        Fruit proxyBanana = (Fruit) Proxy.newProxyInstance(Fruit.class.getClassLoader(), new Class[]{Fruit.class}, new FruitInvocationHandler(banana));
        System.out.println(proxyApple.getClass().getName());
        System.out.println(proxyBanana.getName());
    }
}

Was ist ein Java-Proxy?

Diese Methode dient dazu, einen PeelFruitProxy wie oben erwähnt zu generieren (der Name, den wir sehen, ist natürlich: com.sun ). Proxy.$Proxy0) wird dynamisch generiert, um zu vermeiden, dass es jedes Mal geschrieben werden muss. Dies ist auch der Grund, warum es als dynamischer Proxy bezeichnet wird, da wir jede Klasse zur Laufzeit vertreten können. AOP ist in vielen Programmen auf diese Weise implementiert, aber wir haben einige Merkmale gefunden. Der zweite Parameter von newProxyInstance() ist eine Liste von Schnittstellen.

Da unsere dynamisch generierte Proxy-Klasse auch eine Schnittstelle implementieren muss, damit sie leicht nach unten transformiert und ihre Methoden verwendet werden kann, hat die generierte Klasse sonst einen Klassennamen wie com.sun.proxy.$ Da sich Proxy0 im Speicher befindet, kann die generierte Methode nicht aufgerufen werden. ** Daher weist diese dynamische Proxy-Methode einen schwerwiegenden Mangel auf, nämlich dass die Proxy-Klasse die Schnittstelle implementieren muss. **

CGLib-Proxy

cglib is a powerful, high performance and quality Code Generation Library, It is used to extend JAVA classes and implements interfaces at runtime.

Eine weitere bekannte Java-Proxy-Implementierung ist CGLib (Code Generation Library), ein auf ASM basierendes Codegenerierungs-Framework, das Sie verwenden können Um Klassen dynamisch zu generieren und dann das Abfangen von Methoden zu implementieren, können Sie das Problem des dynamischen Proxys von JDK vermeiden, bei dem die Zielklasse die Schnittstelle implementieren muss. Mit anderen Worten: CGLib kann zum Generieren des oben erwähnten PeelFruitProxy verwendet werden.

Lassen Sie mich kurz vorstellen, wie man es verwendet. Diese CGLib ist eine Bibliothek eines Drittanbieters. Wir müssen uns darauf verlassen:

compile 'cglib:cglib:3.2.8 '


Die neueste Version ist hier zu sehen (neue Version) [https://github.com/cglib/cglib/releases] Dann probieren wir es aus und implementieren den oben genannten Agenten

//FruitMethodInterceptor.java/**
 * CGLib代理的方法拦截器
 */public class FruitMethodInterceptor implements MethodInterceptor{    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        String result = (String) proxy.invokeSuper(obj, args);//主要,这里调用的是父类,也就是说, 生成的类和原始类是继承关系
        return "去皮的"+result;
    }
}//Main.javapublic class Main {    public static void main(String[] args) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(Apple.class);
        enhancer.setCallback(new FruitMethodInterceptor());
        Apple apple = (Apple) enhancer.create();
        System.out.println(apple.getClass().getName());
        System.out.println(apple.getName());
    }
}

run Der Effekt ist wie folgt:

Was ist ein Java-Proxy?

Wir sehen, dass die gleiche Funktion implementiert ist, aber Apple ist nicht mehr die ursprüngliche Apple-Klasse, sondern com .zjiecode.learn.java. Proxy.Apple$$EnhancerByCGLIB$$44ade224, ja, wir verwenden tatsächlich diese Klasse, nicht die ursprüngliche Apple-Klasse, und implementiert schließlich den Proxy für die Apple-Klasse. Da die Vererbung verwendet wird, ist es auf diese Weise nicht erforderlich, dass die Proxy-Klasse die Schnittstelle implementiert. Natürlich kann er Agenten auch über Schnittstellen implementieren.

Zusammenfassung

Der erste Proxy-Typ wird hier nicht erwähnt. Er ist nur für einen einzelnen Schnittstellen-Proxy geeignet und wird zur Kompilierungszeit festgelegt.

Die zweite und dritte Art von Proxys sind alle dynamische Proxys, aber wir sehen, dass es Unterschiede gibt:

1) Der dynamische Proxy von JDK kann nur Schnittstellen-Proxys implementieren und ist ein gepacktes Proxy-Objekt (. Das heißt, während des Proxy-Prozesses gibt es zwei Objekte, ein Proxy-Objekt und ein Zielobjekt. Das Zielobjekt wird in das Proxy-Objekt gepackt.

2) Der Proxy von CGLib erbt das Zielobjekt, generiert eine neue Klasse und implementiert dann den Proxy. Auf diese Weise befindet sich ein Proxyobjekt im Speicher und es wird keine direkte Vererbung verwendet Methode

generiert Proxy-Klassen zur Laufzeit, was sich von Javapoet unterscheidet, das Klassen zur Kompilierungszeit generiert.

Das obige ist der detaillierte Inhalt vonWas ist ein Java-Proxy?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Vorheriger Artikel:Was nützt das in Java?Nächster Artikel:Was nützt das in Java?