动态代理和静态代理的区别

作者:雨夜偷牛的人
链接:https://www.zhihu.com/question/20794107/answer/23330381
来源:知乎
著作权归作者所有,转载请联系作者获得授权。

关于动态代理设计模式很可能题主就在不知不觉中使用了,例如Spring中的AOP,Struts2中的拦截器等。

先来看静态代理模式代码:
package test;

public interface Subject   
{   
  public void doSomething();   
}

package test;

public class RealSubject implements Subject   
{   
  public void doSomething()   
  {   
    System.out.println( "call doSomething()" );   
  }   
}  

package test;

public class SubjectProxy implements Subject
{
  Subject subimpl = new RealSubject();
  public void doSomething()
  {
     subimpl.doSomething();
  }
}

package test;

public class TestProxy 
{
   public static void main(String args[])
   {
       Subject sub = new SubjectProxy();
       sub.doSomething();
   }
}

刚开始我会觉得SubjectProxy定义出来纯属多余,直接实例化实现类完成操作不就结了吗?后来随着业务庞大,你就会知道,实现proxy类对真实类的封装对于粒度的控制有着重要的意义。但是静态代理这个模式本身有个大问题,如果类方法数量越来越多的时候,代理类的代码量是十分庞大的。所以引入动态代理来解决此类问题。

先看代码:
package test;

public interface Subject   
{   
  public void doSomething();   
}

package test;

public class RealSubject implements Subject   
{   
  public void doSomething()   
  {   
    System.out.println( "call doSomething()" );   
  }   
}  

package test;

import java.lang.reflect.InvocationHandler;  
import java.lang.reflect.Method;  
import java.lang.reflect.Proxy;  

public class ProxyHandler implements InvocationHandler
{
    private Object tar;

    //绑定委托对象,并返回代理类
    public Object bind(Object tar)
    {
        this.tar = tar;
        //绑定该类实现的所有接口,取得代理类 
        return Proxy.newProxyInstance(tar.getClass().getClassLoader(),
                                      tar.getClass().getInterfaces(),
                                      this);
    }    

    public Object invoke(Object proxy , Method method , Object[] args)throws Throwable
    {
        Object result = null;
        //这里就可以进行所谓的AOP编程了
        //在调用具体函数方法前,执行功能处理
        result = method.invoke(tar,args);
        //在调用具体函数方法后,执行功能处理
        return result;
    }
}

public class TestProxy
{
    public static void main(String args[])
    {
           ProxyHandler proxy = new ProxyHandler();
           //绑定该类实现的所有接口
           Subject sub = (Subject) proxy.bind(new RealSubject());
           sub.doSomething();
    }
}

看完代码,现在我来回答,动态代理的作用是什么:
  1. Proxy类的代码量被固定下来,不会因为业务的逐渐庞大而庞大;
  2. 可以实现AOP编程,实际上静态代理也可以实现,总的来说,AOP可以算作是代理模式的一个典型应用;
  3. 解耦,通过参数就可以判断真实类,不需要事先实例化,更加灵活多变。
### 动态代理静态代理的主要区别 #### 定义时间不同 静态代理中,代理类由程序员预先定义并编译完成。这意味着在编写应用程序时已经确定了具体的代理逻辑[^1]。 对于动态代理而言,其代理对象是在程序执行期间依据特定指令自动生成的,并不是通过显式的Java源文件来描述这些类型的结构[^2]。 #### 创建灵活性对比 由于静态代理依赖于预定义的具体实现类,因此当需求变化或者需要增加新的功能时,可能不得不修改现有代码甚至重新设计整个体系架构;相反地,借助于JDK内置机制或是第三方库如CGLIB的支持下所构建出来的动态代理方案则允许开发者更加灵活便捷地向已有的业务流程注入额外行为而不必触及原始组件本身[^4]。 #### 性能考量方面 理论上讲,因为不需要即时生成字节码以及解析过程所带来的开销使得采用传统意义上的“硬编码”形式即静态化手段往往可以获得更好的性能表现;不过随着现代虚拟机技术的发展进步特别是Just-In-Time (JIT) 编译器优化措施的应用推广, 这种差距正在逐渐缩小,在大多数实际应用场景里两者间的效率差异几乎可以忽略不计[^3]。 ```java // 静态代理示例 public class StaticProxy implements Service { private final RealService realService; public StaticProxy(RealService realService){ this.realService = realService; } @Override public void execute(){ System.out.println("Before method call"); realService.execute(); System.out.println("After method call"); } } ``` ```java // 使用JDK动态代理的例子 import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; class DynamicProxyExample implements InvocationHandler { private Object target; public static Object newInstance(Object target) { return Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), new DynamicProxyExample(target)); } private DynamicProxyExample(Object target) {this.target = target;} @Override public Object invoke(Object proxy, Method method, Object[] args)throws Throwable{ System.out.println("Before calling " + method.getName()); Object result = method.invoke(target,args); System.out.println("Called " + method.getName() +" successfully."); return result; } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值