Java实现动态代理

1、 简单代理模式的实现。

 

 interface Interface{
 void doSomething();
 void doSomethingElse(String arg);
}
class RealObject implements Interface{
 public void doSomething(){
  System.out.println("doSomething");
 }
 public void doSomethingElse(String arg){
  System.out.println("doSomethingElse");
 }
}
class SimpleProxy implements Interface{
 private Interface proxied;
 public SimpleProxy(Interface proxied){
  this.proxied = proxied;
 }
 public void doSomething() {
  System.out.println("SimpleProxy doSomething");
  proxied.doSomething();
 }

 public void doSomethingElse(String arg) {
  System.out.println("SimpleProxy doSomethingElse");
  proxied.doSomethingElse(arg);
 }
 
}
public class SimpleProxyDemo {
 public static void consumer(Interface inter){
  inter.doSomething();
  inter.doSomethingElse("wooooo");
 }
 public static void main(String[] args){
  consumer(new RealObject());
  consumer(new SimpleProxy(new RealObject()));
 }
}

 

2、动态代理的实现。

class DynamicProxyHandler implements InvocationHandler{
 private Object proxied;
 public DynamicProxyHandler(Object proxied){
  this.proxied = proxied;
 }
 public Object invoke(Object proxy, Method method, Object[] args)
   throws Throwable {
  System.out.println("**** proxy: " + proxy.getClass() + ", mehtod: " + method + ", args: " + args);
  if(args != null)
   for(Object arg : args)
    System.out.println(" " + arg);
  
  return method.invoke(proxied, args);
 }
 
}
public class SimpleDynamicProxy {
 public static void consumer(Interface inter){
  inter.doSomething();
  inter.doSomethingElse("wooooo");
 }
 public static void main(String[] args){
  RealObject real = new RealObject();
  consumer(real);
  Interface proxy = (Interface)Proxy.newProxyInstance(Interface.class.getClassLoader(), new Class[]{Interface.class}, new DynamicProxyHandler(real));
  consumer(proxy);
 }
}

        通过调用静态方法Proxy.newProxyInstance()可以创建动态代理,这个方法需要得到一个类加载器,一个你希望该代理实现的接口列表(不是类或抽象类),以及InvocationHandler接口的一个实现。动态代理可以将所有的调用重定向到调用处理器,因此通常会向调用处理器的构造器传递给一个“实际”对象的引用,从而使得调用处理器在执行其中介任务时,可以将请求转发。

### JAVA动态代理实现方式 Java中的动态代理可以通过两种主要方式实现:JDK原生动态代理和CGLIB动态代理。以下是两种实现方式的详细介绍: #### 1. JDK原生动态代理 JDK动态代理是基于`java.lang.reflect.Proxy`类和`InvocationHandler`接口实现的。它通过反射机制生成一个实现代理接口的代理类,并在调用具体方法时通过`InvocationHandler`来处理。这种方式要求被代理的对象必须实现接口[^1]。 以下是JDK动态代理的基本实现步骤: - 使用`Proxy.newProxyInstance`方法创建代理对象。 - 实现`InvocationHandler`接口,重写`invoke`方法以定义代理逻辑。 **代码示例:** ```java import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; interface Person { void sayHello(); } class RealPerson implements Person { @Override public void sayHello() { System.out.println("Hello, World!"); } } class MyInvocationHandler implements InvocationHandler { private Object target; public MyInvocationHandler(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("Before method call"); Object result = method.invoke(target, args); System.out.println("After method call"); return result; } } public class JdkDynamicProxyExample { public static void main(String[] args) { Person realPerson = new RealPerson(); Person proxyPerson = (Person) Proxy.newProxyInstance( realPerson.getClass().getClassLoader(), realPerson.getClass().getInterfaces(), new MyInvocationHandler(realPerson) ); proxyPerson.sayHello(); } } ``` #### 2. CGLIB动态代理 CGLIB(Code Generation Library)是一个第三方库,通过字节码增强技术生成目标类的子类来实现动态代理。这种方式不需要目标类实现接口,因此适用于没有接口的场景[^5]。 CGLIB的核心类是`Enhancer`,通过设置回调函数(如`MethodInterceptor`)来定义代理逻辑。 **代码示例:** ```java import net.sf.cglib.proxy.Enhancer; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy; import java.lang.reflect.Method; class Actor { public void basicAct(Float money) { System.out.println("Basic act with " + money); } public void advancedAct(Float money) { System.out.println("Advanced act with " + money); } } public class CglibDynamicProxyExample { public static void main(String[] args) { Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(Actor.class); enhancer.setCallback(new MethodInterceptor() { @Override public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { Float money = (Float) args[0]; if ("basicAct".equals(method.getName()) && money > 10000) { return proxy.invokeSuper(obj, args); } else if ("advancedAct".equals(method.getName()) && money > 50000) { return proxy.invokeSuper(obj, args); } return null; } }); Actor proxyActor = (Actor) enhancer.create(); proxyActor.basicAct(15000f); proxyActor.advancedAct(60000f); } } ``` ### 比较两种实现方式 | 特性 | JDK动态代理 | CGLIB动态代理 | |------------------|--------------------------------|-------------------------------| | 是否需要接口 | 需要 | 不需要 | | 实现原理 | 反射机制 | 字节码增强 | | 性能 | 较慢(因反射开销) | 较快 | | 适用场景 | 目标对象有接口 | 目标对象无接口 | ### 动态代理的应用场景 动态代理通常用于AOP(面向切面编程),可以在不修改目标对象源代码的情况下添加额外的功能,例如日志记录、性能监控、事务管理等[^3]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值