代理模式
- AOP的实现机制利用代理实现业务拓展
- 代码:Spring-08-demo2-demo4

静态代理
角色分析:
- 抽象角色:一般使用接口或者抽象类来解决
- 真实角色:被代理的角色
- 代理角色:代理真实角色,代理真实角色后,我们一般会做一些附属操纵
- 客户:访问代理对象的人
代理步骤:
1.接口
public interface Rent { public void rent(); }
public interface UserService { public void add(); public void delete(); public void update(); public void query();
}
|
2.真实角色
public class Host implements Rent{ @Override public void rent() { System.out.println("房东要出租"); } }
public class UserServiceImpl implements UserService{ @Override public void add() { System.out.println("增加了一个用户"); }
@Override public void delete() { System.out.println("删除了一个用户"); }
@Override public void update() { System.out.println("修改了一个用户"); }
@Override public void query() { System.out.println("查询了一个用户"); } }
|
3.代理角色
public class Proxy implements Rent{ private Host host;
public Proxy() { }
public Proxy(Host host) { this.host = host; }
@Override public void rent() { host.rent(); seeHouse; fare; hetong; }
public void seeHouse(){ System.out.println("中介带你看房"); }
public void fare(){ System.out.println("收中介费"); }
public void hetong(){ System.out.println("签租赁合同"); } }
public class UserServiceProxy implements UserService{
private UserServiceImpl userService;
public void setUserService(UserServiceImpl userService) { this.userService = userService; }
@Override public void add() { log("add"); userService.add();
}
@Override public void delete() { log("delete"); userService.delete(); }
@Override public void update() { log("update"); userService.update(); }
@Override public void query() { log("query"); userService.query(); }
public void log(String msg){ System.out.println("使用了"+msg+"方法"); } }
|
4.客户端访问代理角色
public class Client { public static void main(String[] args) { Host host = new Host(); Proxy proxy = new Proxy(host);
proxy.rent(); } }
public class Client { public static void main(String[] args) { UserServiceImpl userService = new UserServiceImpl();
UserServiceProxy proxy =new UserServiceProxy(); proxy.setUserService(userService); proxy.add(); } }
|
代理模式的好处:
- 可以使真实角色的操作更加纯粹! 不用去关注一些公共的业务
- 公共业务就交给角色! 实现业务分工
- 公共业务发生拓展的时候,方便集中管理
缺点:
- 一个真实角色就会产生一个代理角色; 代码量翻倍 ~ 开发效率会变低
动态代理 – (底层:反射)
- 动态代理角色和静态代理角色一样
- 动态代理的代理类是动态生成的,不是我们直接写好的!
- 动态代理分为两大类:基于接口的动态代理,基于类的动态代理
- 基于接口 – JDK 代理 [使用]
- 基于类:cglib
- java字节码实现:JAVAssist
需要了解两个类: Proxy 代理 , InvocationHdandler
动态代理的好处:
- 可以使真实角色的操作更加纯粹!不用去关注一些公共的业务
- 公共也就就交给代理角色!实现了业务的分工!
- 公共业务发生扩展的时候,方便集中管理!
- 一个动态代理类代理的是一个接口,一般就是对应的一类业务

复用静态代理代码2
public class ProxyInvocationHandler implements InvocationHandler {
private Object target;
public void setTarget(Object target) { this.target = target; }
public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this); }
@Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object result = method.invoke(target, args); logcc(method.getName()); return result; }
public void logcc(String msg){ System.out.println("执行了"+msg+"方法"); }
}
|
public class Client { public static void main(String[] args) {
UserServiceImpl userService=new UserServiceImpl();
ProxyInvocationHandler pih = new ProxyInvocationHandler();
pih.setTarget(userService);
UserService proxy = (UserService) pih.getProxy(); proxy.add(); }
}
|
例子
public interface Start {
String sing(String name); void dance(); }
|
package com.chen.Proxy;
public class BigStart implements Start{
private String name;
public BigStart() { }
public BigStart(String name) { this.name = name; }
@Override public String sing(String name) { System.out.println(this.name + " 正在唱 "+name); return "执行跳舞"; }
@Override public void dance() { System.out.println(this.name + " 正在跳舞"); }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
}
|
public class ProxyUtil {
public static Start createProxy(BigStart bigStart){
Start start = (Start) Proxy.newProxyInstance( ProxyUtil.class.getClassLoader(), new Class[]{Start.class}, new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if (method.getName().equals("sing")){ System.out.println("代理角色的附加操作:准备话筒");
}else if (method.getName().equals("dance")){ System.out.println("代理角色的附加操作:准备舞台");
}
return method.invoke(bigStart,args); } } ); return start; } }
|
BigStart bigStart = new BigStart("坤坤"); Start proxy = ProxyUtil.createProxy(bigStart); String singing = proxy.sing("太美"); proxy.dance(); System.out.println(singing);
|
