- 使用:SpringAOP的底层
- 分类:静态代理和动态代理
1、静态代理
角色分析:
- 抽象角色:一般会使用角色或者抽象类来解决
- 真实角色:被代理的角色
- 代理角色:代理真实角色后,一般会做一些附属操作
- 客户:访问代理对象的人
优点:
- 可以使真实角色的操作更加纯粹,不用去关注一些公共的业务
- 公共业务交给代理角色,实现了业务的分工
- 公共业务发生扩展的时候,方便集中管理
缺点:
- 一个真实角色就会产生一个代理角色,代码量会翻倍,开发效率会变低
租客、房东、中介案例:
//租房接口
public interface Rent {
public void rent();
}
//房东要出租房子
public class Host implements Rent{
@Override
public void rent() {
System.out.println("房东要出租房子!");
}
}
//中介在租房子同时作恶一些附加操作
public class Proxy implements Rent{
private Host host;
public Proxy(){}
public Proxy(Host host){
this.host = host;
}
public void rent(){
seeHouse();
host.rent();
hetong();
fare();
}
public void seeHouse(){
System.out.println("中介带你看房");
}
public void hetong(){
System.out.println("签租赁合同");
}
public void fare(){
System.out.println("收中介费");
}
}
//租客租房子
public class Client {
public static void main(String[] args) {
Host host = new Host();
Proxy proxy = new Proxy(host);
proxy.rent();
}
}
2、加深理解
实践代码:
//Service接口层
public interface UserService {
public void add();
public void delete();
public void update();
public void query();
}
//Service接口实现层
public class UserserviceImpl implements UserService{
public void add() {
System.out.println("增加了一个用户");
}
public void delete() {
System.out.println("删除了一个用户");
}
public void update() {
System.out.println("修改了一个用户");
}
public void query() {
System.out.println("查询了一个用户");
}
}
//代理层,主要为了增加输出日志功能
public class UserServiceProxy {
private UserserviceImpl userservice;
public void setUserService(UserserviceImpl 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("[Debug]使用了"+msg+"方法");
}
}
//访问代理层
public class Client {
public static void main(String[] args) {
UserserviceImpl userService = new UserserviceImpl();
UserServiceProxy proxy = new UserServiceProxy();
proxy.setUserService(userService);
proxy.query();
}
}
3、动态代理
- 和静态代理相同及不同点:
- 相同点:和静态代理角色一样
- 代理类是动态生成的,不是直接写好的
- 分类:基于接口的动态代理和基于类的动态代理
- 基于接口:JDK
- 基于类:cglib
- java字节码实现:javasist
- 使用到的类:
- Proxy:代理
- InvocationHandler:调用处理程序
- 优点:
- 可以使真实角色的操作更加纯粹,不用去关注一些公共的业务
- 公共业务交给代理角色,实现了业务的分工
- 公共业务发生扩展的时候,方便集中管理
- 一个动态代理类代理的是一个接口,一般就是对应的一类业务
- 一个动态代理类可以代理多个类,只要是实现了同一个接口
使用:
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);
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
log(method.getName());
//执行target接口中的args方法,返回执行结果
Object result = method.invoke(target, args);
return result;
}
public void log(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.query();
}
}