Was ist ein Java-Proxy?
Was ist ein Proxy? Der Zugriff auf das Ziel erfolgt nicht direkt, sondern über eine Zwischenschicht, wie folgt: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()); } }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 '
//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: 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!