Java静态\动态代理之百炼成仙

Java代理

在代理之前,我们需要知道我们代理的东西是什么

比如说我们有一个原对象,我们不直接去访问他,我们通过代理去访问它,这么做的目的是,以后不用修改原对象,可以通过修改代理类,来实现具体的方法,来看具体例子:

首先要有一个接口

public interface IUser {
    void show();
    
    void create();

    void update();
}

接着实现一下

public class Userimpl implements IUser {
    public Userimpl(){}

    public void show(){
        System.out.println("展示");
    }
    
    public void create(){
        System.out.println("新建");
    }

    public void update(){
        System.out.println("更新");
    }
}

再来一个代理

public class UserProxy implements IUser{
    //这个属于静态代理
    IUser user;
    public UserProxy(){}

    public UserProxy(IUser user){ this.user = user; }

    @Override
    public void show() {
        user.show();
        System.out.println("调用了show");
    }
    public void create(){
        user.create();
        System.out.println("调用了新建");
    }

    public void update(){
        user.update();
        System.out.println("调用了更新");
    }
}

然后我们就可以用了

public class ProxyTest {
    public static void main(String[] args) {
        IUser user = new Userimpl();
//        user.show();
        //静态代理
        IUser userProxy = new UserProxy(user);//用代理类过一遍我们要用的方法,达到一个日志的效果
        userProxy.show();
        
 输出:
     展示
     调用了show

此处这个代理方法与调用本身没有什么关系,可以当成一个记录日志的行为,但是这个行为本身和我调用show方法没什么联系,但是这个日志又必须要有,但是问题就来了,如果我们要代理的东西太多,怎么办??总不能先上面我写的一样,一个方法一个日志记录格式吧,于是java官方便出了另一种代理方式,叫动态代理

这个动态代理我们可以假设一下的工作机制,无非就是为了方便点来做到调用

public class UserProxy implements IUser{

    IUser user;
    public UserProxy(){}
    public UserProxy(IUser user){ this.user = user; }
    
    IUser user = new Userimpl(); 
    Class c = user.getClass();
    //利用反射来获取方法
    Method usermethod = c.getMethod("show");
    
    public void execute(){
        System.out.println("调用了" + usermethod.getName());
		usermethod.invoke(user,args)//因为无参,这里不应该有args
    }
    
}

(上述代码我直接在md敲的,没有导入包)

但是我们发现还有一个问题就方法获取的一个问题,依然冗杂了,如果我们要的不是show,那么就要替换show,或者再来一个反射,但是不巧的是,java官方针对获取方法冗杂这个问题,来了一个很好的解决办法,也就是动态代理,来看一下

java.lang.reflect.Proxy//这里面的一个方法

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

它的前两个参数都可以有固定写法,比较好解决

Proxy.newProxyInstance(user.getClass().getClassLoader(),
                user.getClass().getInterfaces(),
                userinvocationhandler);

但是最后一个处理器参数,我们需要关心一下

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class UserInvocationHandler implements InvocationHandler {
    IUser user;

    public UserInvocationHandler(){}

    public UserInvocationHandler(IUser user){
        this.user = user;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        System.out.println("调用了" + method.getName());
        method.invoke(user,args);
        return null;
    }
}

这样的话main方法就是这样

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

public class ProxyTest {
    public static void main(String[] args) {
        IUser user = new Userimpl();
        //动态代理
        //要代理的接口,要做的事情,classloader
        InvocationHandler userinvocationhandler = new UserInvocationHandler(user);
        IUser userproxyInstance = (IUser) Proxy.newProxyInstance(user.getClass().getClassLoader(),
                new Class[]{IUser.class},
                userinvocationhandler);
        userproxyInstance.update();
    }
}

而我们所选择的method的传递方式就可以是userproxyInstance.方法名了,用起来比较方便

  • 4
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是Java静态代理和动态代理的介绍和示例: 静态代理: 静态代理是在编译期间就已经确定代理类的代码,需要为每一个被代理的类编写一个代理类。静态代理的优点是简单易懂,缺点是需要手动编写代理类,当被代理的类增加方法时,代理类也需要相应地修改。 示例代码: 假设有一个接口`Subject`和它的实现类`RealSubject`,我们需要为`RealSubject`类创建一个代理类`ProxySubject`,并在代理类中调用`RealSubject`类的方法。 ```java // 接口 public interface Subject { void request(); } // 实现类 public class RealSubject implements Subject { @Override public void request() { System.out.println("RealSubject request"); } } // 代理类 public class ProxySubject implements Subject { private RealSubject realSubject; public ProxySubject(RealSubject realSubject) { this.realSubject = realSubject; } @Override public void request() { System.out.println("ProxySubject request"); realSubject.request(); } } // 测试类 public class Test { public static void main(String[] args) { RealSubject realSubject = new RealSubject(); ProxySubject proxySubject = new ProxySubject(realSubject); proxySubject.request(); } } ``` 动态代理动态代理是在运行时动态生成代理类的代码,不需要事先知道代理类的代码。Java中的动态代理主要有两种形式,一种是基于接口的动态代理,另一种是基于类的动态代理。其中,基于接口的动态代理使用较为广泛。 示例代码: 假设有一个接口`Subject`和它的实现类`RealSubject`,我们需要为`RealSubject`类创建一个动态代理类,并在代理类中调用`RealSubject`类的方法。 ```java // 接口 public interface Subject { void request(); } // 实现类 public class RealSubject implements Subject { @Override public void request() { System.out.println("RealSubject request"); } } // 动态代理类 public class DynamicProxy implements InvocationHandler { private Object target; public DynamicProxy(Object target) { this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("DynamicProxy request"); Object result = method.invoke(target, args); return result; } } // 测试类 public class Test { public static void main(String[] args) { RealSubject realSubject = new RealSubject(); DynamicProxy dynamicProxy = new DynamicProxy(realSubject); Subject proxySubject = (Subject) Proxy.newProxyInstance( realSubject.getClass().getClassLoader(), realSubject.getClass().getInterfaces(), dynamicProxy); proxySubject.request(); } } ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Yn8rt

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值