Java的三种代理模式

博客介绍了Java中的代理模式,包括静态代理、动态代理(JDK代理)和cglib代理。指出静态代理存在代理类多、维护不便的缺点,而cglib代理可在内存中构建子类对象实现代理,无需实现接口,还介绍了其实现方法及使用场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Proxy

转载:https://www.cnblogs.com/cenyu/p/6289209.html

代理对象是对目标对象的扩展,并会调用目标对象

1. 静态代理

interface IUserDao(){
    void save();
}

public class UserDao implements IUserDao(){
    public void save(){
        //to save...
    }
}

public class UserDaoProxy implements IUserDao(){
    private UserDao userDao;
    public void save(){
        //开始事务
        userDao.save();
        //提交事务
    }
}

缺点: 因为代理类要和目标类实现一样的接口,所以会有很多代理类,一旦接口新增方法,代理类和目标类都要维护。

2. 动态代理(JDK代理)

1. 特点:不需要实现接口,但需要指定接口类型。目标类必须实现接口。
//IUserDao, UserDao 同上
//代理工厂类
public class ProxyFactory{
    //目标对象
    private Object target;
    public ProxyFactory(Object target){
        this.target=target;
    }
    //给目标对象生成代理对象
    public Obejct getProxyInstance(){
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),
            target.getClass().getInterfaces(),
            new InvocationHandler(){
                @Override
                public Object invoke(Object proxy, Method method, Obejct[] args) throws Throwable{
                    //开始事务
                    //执行目标方法
                    Object value = method.invoke(target,args);
                    //提交事务
                    return value;
                }
            }); 
    }
}
//测试类
public class Test(){
    public static void main(String[] args){
        final IUserDao userDao = (IUserDao) new ProxyFactory(new UserDao()).getProxyInstance();
        System.out.println("proxy class==" + userDao.getClass());
        userDao.save();
    }
}

3. cglib代理

  • 上面的静态代理和动态代理都需要实现接口。而cglib在内存中构建子类对象从而实现对目标类的代理功能。
  • cglib是一个强大的高性能的代码生成包,它可以在运行期扩展java类和实现java接口。它广泛的被许多AOP的框架使用。例如Spring aop 和synaop,为他们提供方法的interception(拦截)。
  • 实现方法:
    1. 引入jar包
    2. 代理的类不能是final
    3. 目标对象的方法如果是final/static,那就就不会被拦截。
//目标类
public class UserDao(){
    public void save(){
        //todo 
    }
}

public class CglibProxyFactory implements MethodInterceptor {

    private Object target;

    public CglibProxyFactory(Object target) {
        this.target = target;
    }

    //创建代理对象
    public Object getProxyInstance() {
        Enhancer en = new Enhancer();
        en.setSuperclass(target.getClass());
        //设置回调函数
        en.setCallback(this);
        //创建子类(代理对象)
        return en.create();
    }

    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        //开始事务
        System.out.println("---- transaction begin ----");
        //执行目标方法
        Object value = method.invoke(target, objects);
        //提交事务
        System.out.println("---- transaction end ----");
        return value;
    }
}

//测试类
public class Test(){
    public static void main(String[] args){
        UserDao userDao = (UserDao) new ProxyFactory(new UserDao()).getProxyInstance();
        System.out.println("proxy class==" + userDao.getClass());
        userDao.save();
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值