代理模式
代理模式
代理模式中,代理类与主体类实现同样的接口,代理类持有实体类的引用,并接受客户端对代理类中实体引用的外部注入,并代理实体类的功能。
注:描述中的这种外部注入形式有个专有技术名词:依赖注入

静态代理模式
角色
- 抽象角色:一般会使用接口或者抽象类来解决
- 真实角色:被代理的角色
- 代理角色:代理真实角色,一般会有一些附属操作
- 客户:访问代理对象的角色
优点
- 可以使真实角色更加集中本身的业务,不用去关注一些公共的业务
- 公共业务交给代理角色,实现业务的分工
- 公共业务发生扩展时,方便集中管理
缺点
每多一个真实角色就会产生一个代理角色:代码量会翻倍,开发效率低
代码步骤
1.接口
public interface UsersService {
public void add();
public void delete();
public void update();
public void query();
}
2.真实角色
public class UserServiceImp implements UsersService{
public void add() {
System.out.println("添加");
}
public void delete() {
System.out.println("删除");
}
public void update() {
System.out.println("修改");
}
public void query() {
System.out.println("查询");
}
}
3.代理角色
public class UserServiceProxy implements UsersService{
private UserServiceImp userService;
public void setUserService(UserServiceImp userService) {
this.userService = userService;
}
public void add() {
log("add");
userService.add();
}
public void delete() {
log("delete");
userService.delete();
}
public void update() {
log("update");
userService.update();
}
public void query() {
log("query");
userService.query();
}
public void log(String msg){
System.out.println("使用了"+msg+"方法");
}
}
4.客户访问代理角色
public class User {
public static void main(String[] args) {
UserServiceImp userService = new UserServiceImp();
UserServiceProxy proxy = new UserServiceProxy();
proxy.setUserService(userService);
proxy.add();
}
}
运行结果:
使用了add方法
添加
Process finished with exit code 0
动态代理模式
- 和静态代理模式中的角色一样
- 动态代理的具体代理类的生成是在运行期动态产生的,而非编译期就已经静态存在;
- 动态代理分为:
- 基于接口:JDK动态代理
- 基于类:CGLIB动态代理
- Java字节码实现:javassist
JDK中的动态代理是通过反射类Proxy以及InvocationHandler回调接口实现的;但是,JDK中所要进行动态代理的类必须要实现一个接口,也就是说只能对该类所实现接口中定义的方法进行代理,这在实际编程中具有一定的局限性,而且使用反射的效率也并不是很高。
JDK动态代理核心代码:
1.实现这个接口InvocationHandler,创建代理对象
2.Proxy提供创建动态代理类和实例的静态方法:Proxy.newProxyInstance()
//自动生成代理类
public class DynamicProxy implements InvocationHandler {
//被代理的接口
private Object target;
//set方法
public void setTarget(Object target) {
this.target = target;
}
//生成得到代理类
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
target.getClass().getInterfaces(),this);
}
//处理代理实例并返回结果
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
log(method.getName());
Object result = method.invoke(target,args);
return result;
}
//代理角色的附属业务的例子
public void log(String msg){
System.out.println("使用了"+msg+"方法");
}
}
target可表示任务场景中的真实角色,这是一个模板!
客户实现业务:
public class User {
public static void main(String[] args) {
//真实角色
UserServiceImp userService = new UserServiceImp();
//代理角色,因为是动态代理所以不存在
DynamicProxy dp = new DynamicProxy();
//设置代理对象
dp.setTarget(userService);
//动态生成代理类
UsersService proxy =(UsersService) dp.getProxy();
//调用真实角色中的方法
proxy.delete();
}
}