java中的代理模式

java中的代理模式:

定义:给目标对象提供一个代理对象,并且由代理对象控制对目标对象的引用
目的:①:通过代理对象的方式间接的访问目标对象,防止直接访问目标对象给系统带来不必要的复杂性
②:通过代理业务对原有业务进行增强
java当中有三种方式来创建代理对象:静态代理,基于jdk(接口)的动态代理,基于CGLLIB(父类)的动态代理。

静态代理: 在代码的编译阶段植入Pointcut的内容,性能好,需要额外的编译
动态代理:在代码执行阶段,在内存中截获对象,动态的插入Pointcut的内容,不需要额外的编译,性能比静态织入要低

静态代理

业务接口:

package proxy;

/**
 * Created by YuanXiang on 2017/9/2.
 */
public interface UserManager {

    void addUser(String userId, String userName);

    void delUser(String userId);

    String findUser(String userId);

    void modifyUser(String userId, String userName);
}

被代理类:


package proxy;

/**
 * Created by YuanXiang on 2017/9/2.
 */
public class UserManagerImpl implements UserManager {

    @Override
    public void addUser(String userId, String userName) {
        System.out.println("UserManagerImpl.addUser");
    }

    @Override
    public void delUser(String userId) {
        System.out.println("UserManagerImpl.delUser");
    }

    @Override
    public String findUser(String userId) {
        System.out.println("UserManagerImpl.findUser");
        return "张三";
    }

    @Override
    public void modifyUser(String userId, String userName) {
        System.out.println("UserManagerImpl.modifyUser");

    }
}

代理类:


package proxy;

/**
 * Created by YuanXiang on 2017/9/2.
 */
public class UserManagerImplStaticProxy implements UserManager {

    private UserManager userManager;

    public UserManagerImplStaticProxy(UserManager userManager){
        this.userManager = userManager;
    }

    @Override
    public void addUser(String userId, String userName) {
        System.out.print("start ---");
        this.userManager.addUser(userId, userName);
        System.out.print("end --");
    }

    @Override
    public void delUser(String userId) {

    }

    @Override
    public String findUser(String userId) {
        return null;
    }

    @Override
    public void modifyUser(String userId, String userName) {

    }
}

调用


package proxy;

/**
 * Created by YuanXiang on 2017/9/2.
 */
public class Client {
    public static void main(String[] args) {
        UserManager proxy = new UserManagerImplStaticProxy(new UserManagerImpl());
        proxy.addUser("312","123");
    }
}

静态代理存在哪些问题?
违反了开闭原则:
程序对外扩展开放,对修改关闭,换句话来说,当需求发生变化时,我们可以增加新模块来解决新需求,而不是通过该变原来的代码来解决我们的新需求

动态代理

静态代理会为每一个业务增强都提供一个代理类, 由代理类来创建代理对象, 而动态代理并不存在代理类, 代理对象直接由代理生成工具动态生成.

看完静态代理以及动态代理可以发现,静态代理一般一个代理类只能代理一个类,而动态代理则不同,上述的日志动态代理类可以把代理任何类,把日志输出到任何需要的地方。但是两者都是在面向接口编程的基础之上。
(1) 转账业务

public interface IAccountService {
    //主业务逻辑: 转账
    void transfer();
}
public class AccountServiceImpl implements IAccountService {
    @Override
    public void transfer() {
        System.out.println("调用dao层,完成转账主业务.");
    }
}

(2) 增强

因为这里没有配置切入点, 称为切面会有点奇怪, 所以称为增强.

public class AccountAdvice implements InvocationHandler {
    //目标对象
    private IAccountService target;

    public AccountAdvice(IAccountService target) {
        this.target = target;
    }

    /**
     * 代理方法, 每次调用目标方法时都会进到这里
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        before();
        return method.invoke(target, args);
    }

    /**
     * 前置增强
     */
    private void before() {
        System.out.println("对转账人身份进行验证.");
    }
}
(3) 测试
```java
public class Client {
    public static void main(String[] args) {
        //创建目标对象
        IAccountService target = new AccountServiceImpl();
        //创建代理对象
        IAccountService proxy = (IAccountService) Proxy.newProxyInstance(target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                new AccountAdvice(target)
        );
        proxy.transfer();
    }
}
结果: 
对转账人身份进行验证.
调用dao层,完成转账主业务.

https://blog.youkuaiyun.com/litianxiang_kaola/article/details/85335700

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值