百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

一直不明白 InvocationHandler#invoke() 中 proxy 参数有什么作用?

yuyutoo 2024-11-12 14:58 9 浏览 0 评论

故事开始

一直不明白 Java InvocationHandler 类的 invoke() 方法中 proxy 参数有什么作用呢?

先来看一段代码,如下所示:

public class DynamicProxyHandler implements InvocationHandler {
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
    // ...doSomething...
  }
}

疑问:有人知道,在 invoke() 方法中的第一个入参 proxy 有什么作用嘛?一直都没搞懂这个参数有啥用,今天我就要发掘出属于它的其中一个作用。让我们开始吧!

接下来我们将创建一个动态代理,它不仅记录方法调用,还能根据代理对象的不同执行不同的逻辑,代码如下。

步骤 1: 定义接口[玫瑰]

首先,我们定义两个接口,每个接口包含一个方法。

public interface GreetingService {
    void sayHello(String name);
}

public interface FarewellService {
    void sayGoodbye(String name);
}

步骤 2: 实现 InvocationHandler[玫瑰]

接下来,我们创建一个 InvocationHandler 实现,它将根据代理对象的不同执行不同的逻辑。

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

public class DynamicProxyHandler implements InvocationHandler {
    private final Object target;

    public DynamicProxyHandler(Object target) {
        this.target = target;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 根据代理对象的类型执行不同的逻辑
        if (proxy instanceof GreetingService) {
            System.out.println("GreetingService proxy used");
        } else if (proxy instanceof FarewellService) {
            System.out.println("FarewellService proxy used");
        }

        System.out.println("Method " + method.getName() + " is called with arguments " + args);
        // 调用实际对象的方法
        Object result = method.invoke(target, args);
        System.out.println("Method " + method.getName() + " returned " + result);
        return result;
    }
}

步骤 3: 创建代理对象[玫瑰]

现在,我们创建两个代理对象,一个用于 GreetingService,另一个用于 FarewellService

public class ProxyDemo {
    public static void main(String[] args) {
        // 实际对象
        GreetingService realGreetingService = (name) -> System.out.println("Hello " + name);
        FarewellService realFarewellService = (name) -> System.out.println("Goodbye " + name);

        // 创建 GreetingService 代理
        GreetingService proxyGreetingService = (GreetingService) Proxy.newProxyInstance(
                GreetingService.class.getClassLoader(),
                new Class<?>[]{GreetingService.class},
                new DynamicProxyHandler(realGreetingService)
        );

        // 创建 FarewellService 代理
        FarewellService proxyFarewellService = (FarewellService) Proxy.newProxyInstance(
                FarewellService.class.getClassLoader(),
                new Class<?>[]{FarewellService.class},
                new DynamicProxyHandler(realFarewellService)
        );

        // 通过代理调用方法
        proxyGreetingService.sayHello("Kimi");
        proxyFarewellService.sayGoodbye("Kimi");
    }
}

程序输出[玫瑰]

当你运行 ProxyDemo 类时,输出将如下所示:

GreetingService proxy used
Method sayHello is called with arguments [Kimi]
Hello Kimi
Method sayHello returned null
FarewellService proxy used
Method sayGoodbye is called with arguments [Kimi]
Goodbye Kimi
Method sayGoodbye returned null

最后来个小总结[玫瑰]

在我们的上述这个例子中:

  • GreetingServiceFarewellService 是两个不同的接口。
  • DynamicProxyHandlerInvocationHandler 的实现,它根据代理对象的类型执行不同的逻辑。
  • ProxyDemomain 方法中,我们创建了两个不同的代理对象,一个用于 GreetingService,另一个用于 FarewellService
  • 我们使用 Proxy.newProxyInstance 方法创建了两个代理对象,它们都使用同一个 DynamicProxyHandler 实例,但是 DynamicProxyHandler 能够根据 proxy 参数的不同区分代理对象的类型。
  • 当我们通过代理对象调用方法时,DynamicProxyHandlerinvoke 方法被调用,它根据 proxy 参数的类型记录了不同的信息,并执行了相应的方法调用。

展示了 proxy 参数如何用于区分不同的代理对象,并允许 InvocationHandler 根据代理对象执行特定的逻辑。通过检查 proxy 参数的类型,DynamicProxyHandler 能够决定如何响应方法调用,从而实现了基于代理对象类型的动态行为。

那么,最后,你还发现 proxy 入参还有哪些用法呢?可以Q在评论区哦!




相关推荐

对volatile,synchronized,AQS的加锁解锁原理的一些理解

一、为什么要加锁,要实现同步多线程编程中,有可能会出现多个线程同时访问同一个共享、可变资源的情况,这个资源我们称之其为临界资源;这种资源可能是:对象、变量、文件等。...

注意,不能错过的CAS+volatile实现同步代码块

前言:最近看到有人说可以使用CAS+volatile实现同步代码块。心想,确实是可以实现的呀!因为AbstractQueuedSynchronizer(简称AQS)内部就是通过CAS+...

面试并发volatile关键字时,我们应该具备哪些谈资?

提前发现更多精彩内容,请访问https://dayarch.top/提前发现更多精彩内容,请访问https://dayarch.top/提前发现更多精彩内容,请访问https://dayarch...

无锁同步-JAVA之Volatile、Atomic和CAS

1、概要本文是无锁同步系列文章的第二篇,主要探讨JAVA中的原子操作,以及如何进行无锁同步。关于JAVA中的原子操作,我们很容易想到的是Volatile变量、java.util.concurrent....

C/C++面试题(二):std::atomic与volatile

volatile是C/C++中的一个关键字,用于告知编译器某个变量的值可能会在程序的控制之外被意外修改(例如被硬件、中断服务程序、多线程环境或其他外部代理)。为了防止编译器对代码进行某些可能破坏...

VOCs(Volatile Organic Compounds)挥发性有机化合物及测试方法

经常看到一些三防漆、涂料、油漆类产品的介绍中提到VOC、VOCs等概念,那么什么是VOC、VOCs和TVOC,VOCs主要包括哪些物质?VOCs的来源有哪些?VOCs的危害及国家标准是什么?一、V...

对volatile 及happen—before的理解

happen—before规则介绍Java...

这一篇我们来了解Synchronized、Volatile、Final关键字

题外话:蓝银王觉醒了!!--来自于一个斗罗大陆动漫爱好者(鹅,打钱!)湿兄这两天回家了,办了点大事,回来的时候我弟弟还舍不得我,哭着不愿意让我回京(我弟还是小学),我也心里很不舍,但是还是要回京奋斗...

关于 Java 关键字 volatile 的总结

1什么是volatilevolatile是Java的一个关键字,它提供了一种轻量级的同步机制。相比于重量级锁synchronized,volatile更为轻量级,因为它不会引起线程上下文...

大白话聊聊Java并发面试问题之volatile到底是什么?

用最简单的大白话,加上多张图给大家说一下,volatile到底是什么?...

为什么要有volatile关键字(volatile 关键字为什么不能保证原子性)

在嵌入式编程和多线程编程中,我们常会见到volatile关键字声明的变量。下面说一下volatile关键字的作用:1.保持变量内存可见简而言之就是用volatile声明的变量会告诉编译器和处理器,这个...

Java的volatile到底怎么理解?(java volatitle)

我们都知道,在Java中有很多的关键字,比如synchronize比如volatile,这些都是一些比较关键的,还有final,今天我们就来聊一下这个volatile因为这个vo...

Java多线程编程中的volatile关键字:解密神秘的共享内存

Java多线程编程中的volatile关键字:解密神秘的共享内存在Java多线程编程的世界里,volatile关键字就像一位低调却至关重要的守护者。它默默无闻地站岗放哨,确保多个线程之间能够正确地共享...

你了解volatile关键字的作用吗?(关键字volatile有什么含意?并举出三个不同的例子?)

【死记硬背】volatile关键字主要用于保持内存的变量可见性和禁止重排序。变量可见性:当一个线程改变了变量的值,那么新的值对于其他线程也是可以立即获取到的。禁止重排序:...

谈谈你对volatile 关键字作用和原理的理解

一位6年工作经验的小伙伴,在某里二面的时候被问到“volatile”关键字。然后,就没有然后了…同样,还有一位4年的小伙伴,去某团面试也被问到“volatile关键字“。然后,也没有然后了…...

取消回复欢迎 发表评论: