【java】代理

什么是代理

假设有一个核心方法叫转账,为了安全性考虑,不能让用户直接访问接口。此时涉及到了代理,这使得用户只能访问代理类,增加了访问限制。

代理的定义:给目标对象提供一个代理对象,并且由代理对象控制着对目标对象的引用。

代理如何实现
基于JDK的代理
① 静态代理

由代理类代理目标类:

目标类生成一个目标对象,代理类首先会生成代理对象,代理对象代理目标对象,普通用户只能访问代理对象。

一个类中可能有很多方法,代理类怎么知道代理目标类的哪个方法呢?除了直接写过去,我们还可以在接口 当中定义目标类当中需要被代理的方法。让目标类实现一个接口,同时代理类也去实现这个接口。

注意:实现接口的是类,而用户访问的是对象。

为什么有接口:接口定义的方法其子类必须实现**,**让目标类实现在接口当中定义的核心方法(目标类实现了接口),为了通知代理类代理目标类当中的哪个核心方法,我们也让代理类实现接口。

在具体实现的过程中,为了保证职能的单一性,功能增强部分的代码最好不写在目标类里。

如何实现静态代理
java 复制代码
public interface ByClothes {  //代理类和目标类都要继承该接口
    //在接口当中定义需要代理的方法
    public void clothes(String size) ;
}
java 复制代码
public class ClothesFactory implements ByClothes{  //目标类 :制作衣服
    public void clothes(String size) {  //目标类的核心
        System.out.println("已经为您定制好了一件大小为" + size + "的衣服");
    }
}
java 复制代码
public class StaticProxy implements ByClothes{  //代理类
    private ClothesFactory clothesFactory = new ClothesFactory();
    //这个地方就是核心方法
    @Override
    public void clothes(String size) {
        front();  //功能增强
        clothesFactory.clothes(size);  //核心方法
        end();  //功能增强
    }
    public void front() {
        System.out.println("市场调研");
    }
    public void end() {
        System.out.println("售后服务");
    }
}
java 复制代码
public class Test { //用户
    public static void main(String[] args) {
        StaticProxy proxy = new StaticProxy();
        proxy.clothes("XXXL");
    }
}

运行结果

② 动态代理

动态代理指的是一个代理类代理多个目标类。

如何实现?类是创建对象的模板

一个代理类生成多个代理对象即可。

为了实现动态化,我们要想办法告诉代码想要代理的是什么。

如何实现动态代理
java 复制代码
public interface ByClothes {  //代理类和目标类都要继承该接口
    //在接口当中定义需要代理的方法
    public void clothes(String size) ;
}
java 复制代码
public interface ByShoes {
    public void shoes(Integer size); //接口当中不需要具体实现
}
java 复制代码
public class ClothesFactory implements ByClothes{  //目标类 :制作衣服
    public void clothes(String size) {  //目标类的核心
        System.out.println("已经为您定制好了一件大小为" + size + "的衣服");
    }
}
java 复制代码
public class ShoesFactory implements ByShoes{
    @Override
    public void shoes(Integer size) {
        System.out.println("已经为您定制了大小为" + size + "码的鞋子");
    }
}
java 复制代码
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class DTProxy implements InvocationHandler {
    //被代理的对象
    private Object factory;
    public DTProxy(Object factory) {
        this.factory = factory;
    }
    //为了能够动态的去实现新增的接口
    //factory.getClass() ---------> 对象名.getClass() == class.forName("全类名")  //生成代理对象
    public Object getProxyInstance() {
        return Proxy.newProxyInstance(factory.getClass().getClassLoader(), factory.getClass().getInterfaces(), this);
    }
    //我们会根据getProxyInstance()方法中所得到的接口信息,得到核心方法
    //我们只需要用反射的方式执行这些方法
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        method.invoke(factory, args);
        return null;
    }
}
java 复制代码
public class Test {
    public static void main(String[] args) {
        ClothesFactory clothesFactory = new ClothesFactory();
        DTProxy p1 = new DTProxy(clothesFactory);
        ByClothes c1 = (ByClothes) p1.getProxyInstance();
        c1.clothes("XXL");

        ShoesFactory shoesFactory = new ShoesFactory();
        DTProxy p2 = new DTProxy(shoesFactory);
        ByShoes c2 = (ByShoes) p2.getProxyInstance();
        c2.shoes(44);
    }
}

运行结果

相关推荐
Blossom.1181 分钟前
使用Python和Scikit-Learn实现机器学习模型调优
开发语言·人工智能·python·深度学习·目标检测·机器学习·scikit-learn
Mr_Air_Boy28 分钟前
SpringBoot使用dynamic配置多数据源时使用@Transactional事务在非primary的数据源上遇到的问题
java·spring boot·后端
曹勖之31 分钟前
基于ROS2,撰写python脚本,根据给定的舵-桨动力学模型实现动力学更新
开发语言·python·机器人·ros2
豆沙沙包?1 小时前
2025年- H77-Lc185--45.跳跃游戏II(贪心)--Java版
java·开发语言·游戏
军训猫猫头1 小时前
96.如何使用C#实现串口发送? C#例子
开发语言·c#
年老体衰按不动键盘1 小时前
快速部署和启动Vue3项目
java·javascript·vue
咖啡啡不加糖1 小时前
Redis大key产生、排查与优化实践
java·数据库·redis·后端·缓存
liuyang-neu2 小时前
java内存模型JMM
java·开发语言
UFIT2 小时前
NoSQL之redis哨兵
java·前端·算法
刘 大 望2 小时前
数据库-联合查询(内连接外连接),子查询,合并查询
java·数据库·sql·mysql