什么是java代理

什么是java代理

代理是设计模式的一种,其原理就是通过代理对象去访问目标对象,而外部只能访问到代理对象

在编程中有这么一种思想,你不要随便的去修改已经写好的代码,如需修改,那么可以通过代理来扩展类的功能。

静态代理

什么是静态代理,静态代理就是你定义一个接口或者是父类,然后代理类与被代理类都需要继承这个接口,在代理类中实现一个方法,需要注入被代理类,这样就能通过代理类去访问被代理类,同时,也可以对被代理类进行增强。

上代码:

/**
 * 定义一个接口,包含一个方法,吃
 */
public interface Person {

    void eat();
}
/**
 * 实现 Person
 * 这个是一个实际的目标类,当前类实现一个吃的方法
 */
public class Man implements Person {
    public void eat() {
        System.out.println("吃饭");
    }
}

/**
 * Man对象的代理,当前代理持有一个Man的对象
 */
public class ManProxy implements Person {

    private Person target;
    public ManProxy(Person target){
        this.target = target;
    }
    public void eat() {
        System.out.println("吃冷菜");
        target.eat();
        System.out.println("吃甜品");
    }
}

/**
 * 测试
 */
public class App {
    public static void main(String[] args) {
        Person target = new Man();
        Person manProxy = new ManProxy(target);
        manProxy.eat();
    }
}

结果:

吃冷菜
吃饭
吃甜品

以上就是一个典型的静态代理的例子,在当前的例子里,有一个Person的接口,该接口的主要作用是定义一个代理类和被代理类都拥有的行为,当前的行为为eat

代理类和被代理类都实现了Person,不过不同的是,代理类持有了一个被代理类的对象,在代理类中可以执行被代理类的方法,因此在代理类中可以在该方法的基础上进行扩展。

缺点:由于代理类需要实现与被代理类同一个接口,那么不同的被代理类就需要去实现不同的接口,那么就会出现代理类很难复用的情况,相对的维护成本也难以控制。

动态代理

什么是动态代理,动态代理就是利用JDK的API动态的在内存中构建代理对象,因此,动态代理也叫做JDK代理,或者接口代理,在动态代理中,代理对象不需要实现接口,但是被代理对象还是需要实现对象的。

生成代理对象的API

生成代理对象的API主要的包在java.lang.reflect.Proxy

static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )

参数说明:
ClassLoader loader : 该参数是被代理对象的使用的类加载器,这个我们直接通过被代理对象就可以获取

Class<?>[] interfaces : 被代理对象实现的接口,这个也可以通过被代理对象获取

InvocationHandler h : 当我们在执行被代理对象的方法的时候,这个处理器就会被执行,当这个方法被执行的时候,会将被代理对象,方法,参数都传入进去

上代码:

/**
 * 代理工厂
 * 主要作用是生成代理对象
 */
public class ProxyFactory {
    //被代理对象
    private Object target;
    public ProxyFactory(Object target){
        this.target = target;
    }

    //为被代理对象生成代理对象
    public Object getProxyInstance(){
        return Proxy.newProxyInstance(
                //指定当前被代理对象使用的类加载器
                target.getClass().getClassLoader(),
                //指定被代理对象实现的接口类型,这里是一个数组,代理可能实现多个接口
                target.getClass().getInterfaces(),
                //事件处理,当我们在执行被代理对象的方法的时候,会触发这个处理器,会把当前执行的被代理对象的方法作为参数传入进去
                new InvocationHandler() {
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        System.out.println("执行前置方法");
                        //执行目标方法
                        Object returnValue = method.invoke(target, args);
                        System.out.println("执行后置方法");
                        return returnValue;
                    }
                }
        );
    }
}
public class App {
    public static void main(String[] args) {
        //被代理对象
        Person target = new Man();
        System.out.println("被代理对象打印:"+target.getClass());
        //创建代理对象
        Person proxy = (Person) new ProxyFactory(target).getProxyInstance();

        System.out.println("代理对象打印:"+proxy.getClass());

        proxy.eat();

    }
}

结果:

被代理对象打印:class com.luban.aop.proxy.statics.Man
代理对象打印:class com.sun.proxy.$Proxy0
执行前置方法
吃饭
执行后置方法

缺点:在动态代理中,代理对象不需要再实现接口了,但是被代理对象还是需要实现接口,否则就无法使用动态代理

Cglib代理

什么是cglib代理,无论是动态代理还是静态代理,都要求我们的被代理的对象必须要实现某一个接口,但是也可能存在这样的情况,一个类没有实现任何的接口,这个时候就可以使用cglib代理。

cglib代理是通过被代理对象的子类的形式来实现代理的,他会在内存中构建一个被代理对象的子类对象,来实现对被代理对象的功能的扩展。

在cglib的底层是通过一个字节码处理框架ASM来转换字节码生成新的类

如果需要使用cglib的话,需要引入一个包 cglib

<!-- https://mvnrepository.com/artifact/cglib/cglib -->
<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>3.3.0</version>
</dependency>

上代码

/**
 * 被代理类
 */
public class Woman {
    public void eat(){
        System.out.println("吃饭");
    }
}

/**
 * cglib 代理工厂类
 */
public class ProxyFactory implements MethodInterceptor {
    //被代理对象
    private Object target;

    public ProxyFactory(Object target){
        this.target = target;
    }
    //为被代理对象创建一个代理对象
    public Object getProxyInstance(){
        //这个是工具类
        Enhancer en = new Enhancer();
        //将被代理类设置为父类
        en.setSuperclass(target.getClass());
        //设置回调
        en.setCallback(this);
        //创建子类(代理对象)
        return en.create();
    }

    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("前置执行....");
        //执行被代理类的方法
        Object returnValue = method.invoke(target,objects);
        System.out.println("后置执行....");
        return returnValue;
    }
}

public class App {
    public static void main(String[] args) {
        //被代理对象
        Woman target = new Woman();
        //代理对象
        Woman proxy = (Woman) new ProxyFactory(target).getProxyInstance();
        //执行方法
        proxy.eat();
    }
}

执行结果:

前置执行....
吃饭
后置执行....

注意:被代理对象的方法不能用final或者static

在springAOP中,如果被代理对象没有实现接口则使用cglib,如果实现了接口则使用动态代理

免费在线视频学习,请访问(持续更新中):java 教程 - 免费分享的视频教程

更多的资讯请访问QQ群:807167416

  • 8
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值