JAvaダイナミックエージェント(jdk&cglib)
4106 ワード
スタティツクエージェント
エージェントクラスと被エージェントクラスは同じインタフェースを実現する
欠点は、エージェントクラスが1つのインタフェースにしか対応していないことです.
ダイナミックエージェント
動的エージェントは2つに分かれています
jdk and
cglib
jdk
jdkエージェントは主に
インタフェースInvocationHandlerこのインタフェースには1つの方法しかありません(コードは次のとおりです)
InvocationHandlerの実装クラスは,具体的なエージェント実装として理解できる.
クラスProxy
エージェントの具体的な操作クラスを生成し、1つのor複数のインタフェースに対して動的にエージェントクラスを実現することができる.
欠点はエージェントされるクラスがインタフェースの実装クラス(インタフェースに依存)でなければならないことである.
一部のクラスがインタフェースを実装していない場合、jdkエージェントは使用できません.
cglib
原理はtargetクラスに対してサブクラスオーバーライドを生成する方法を強化することである.
欠点は継承に基づいてfinalクラスを代理できない(finalクラスはStringなどの継承できない)
必要なjarパッケージ:asm-3.3.1,cglib-2.2.JAr ps:jarパッケージのバージョンが異なるとエラーが発生する可能性があります
エージェントクラスと被エージェントクラスは同じインタフェースを実現する
欠点は、エージェントクラスが1つのインタフェースにしか対応していないことです.
public interface Sale {
public int saleGoods();
}
public class Factory implements Sale {
@Override
public int saleGoods() {
System.out.println("2 , ");
return 2;
}
}
public class Store implements Sale {
private Factory f;
public Store(Factory f) {
this.f = f;
}
@Override
public int saleGoods() {
int price = f.saleGoods();
System.out.println("10 , ");
return price + 8;
}
}
public class Test {
public static void main(String[] args) {
Factory f = new Factory();
Store s = new Store(f);
s.saleGoods();
}
}
ダイナミックエージェント
動的エージェントは2つに分かれています
jdk and
cglib
jdk
jdkエージェントは主に
インタフェースInvocationHandlerこのインタフェースには1つの方法しかありません(コードは次のとおりです)
InvocationHandlerの実装クラスは,具体的なエージェント実装として理解できる.
クラスProxy
エージェントの具体的な操作クラスを生成し、1つのor複数のインタフェースに対して動的にエージェントクラスを実現することができる.
欠点はエージェントされるクラスがインタフェースの実装クラス(インタフェースに依存)でなければならないことである.
一部のクラスがインタフェースを実装していない場合、jdkエージェントは使用できません.
public interface Sale {
public int saleGoods();
}
public class Factory implements Sale {
int price;
@Override
public int saleGoods() {
System.out.println("2 , ");
price = 2;
return price;
}
}
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyStore implements InvocationHandler {
Object factory;
public Object bind(Object factory) {
this.factory = factory;
return Proxy.newProxyInstance(factory.getClass().getClassLoader(),
factory.getClass().getInterfaces(), this);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println("before proxy");
Object result = method.invoke(factory, args);
System.out.println("after proxy");
return result;
}
}
public class Test {
public static void main(String[] args) {
Factory f = new Factory();
Store s = new Store(f);
s.saleGoods();
}
}
cglib
原理はtargetクラスに対してサブクラスオーバーライドを生成する方法を強化することである.
欠点は継承に基づいてfinalクラスを代理できない(finalクラスはStringなどの継承できない)
必要なjarパッケージ:asm-3.3.1,cglib-2.2.JAr ps:jarパッケージのバージョンが異なるとエラーが発生する可能性があります
public class Factory {
int price;
public int saleGoods(int price) {
System.out.println(price + " , ");
this.price = price = 2;
return price;
}
}
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class ProxyStoreCgLib implements MethodInterceptor {
Object target;
public Object getInstanceByMe(Object target) {
this.target = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
enhancer.setCallback(this);
return enhancer.create();
}
@Override
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable {
System.out.println("before");
proxy.invokeSuper(obj, args);
System.out.println("after");
return null;
}
}
public class TestPorxy {
public static void main(String[] args) {
Factory factory = new Factory();
ProxyStoreCgLib proxyCg = new ProxyStoreCgLib();
Factory factoryProxy = (Factory) proxyCg.getInstanceByMe(factory);
factoryProxy.saleGoods(10);
}
}