Java JMX扩展

简介: JMX(Java Management Extensions,即Java管理扩展)是一个为应用程序、设备、系统等植入管理功能的框架。JMX可以跨越一系列异构操作系统平台、系统体系结构和网络传输协议,灵活的开发无缝集成的系统、网络和服务管理应用。

作者: 西魏陶渊明
博客: https://blog.springlearn.cn/

JMX(Java Management Extensions,即Java管理扩展)是一个为应用程序、设备、系统等植入管理功能的框架。JMX可以跨越一系列异构操作系统平台、系统体系结构和网络传输协议,灵活的开发无缝集成的系统、网络和服务管理应用。

前面是对JMX的介绍,那么JMX在我们日常的开发过程中,有什么实际的意义呢? 相信很多做Java开发的同学都使用过JDK自带的 jconsole 或者 jvisualvm 监控过JVM的运行情况,但不知道有没有留意过它们会有一个MBean的功能/标签,通过MBean可以看到在JVM中运行的组件的一些属性和操作。下面小编就通过一个SpringBoot应用来一探究竟。并教会你如何自定义扩展。

## 一、实际意义

### 1. 启动一个SpringBoot应用

下面我们以SpringBoot应用为例子,启动一个SpringBoot项目。端口是 8080

### 2. 命令行打开Jconsole

### 3. 连接前面的应用

image-20200531184151871

选中MBean标签,然后可以看到一个SpringApplication的类。shutdown是服务下线。

当我们点击了shutdown方法后,应用就会自动的关闭了。导致Jconsole连接丢失
image-20200531184702204

getProperty方法是获取应用中的配置信息。如图我们获取redis的相关信息。可以看到返回值是Spring应用中我们定义的值
6379

image-20200531184553346

那么其实这个能力就是利用JMX提供的接口来实现的。下面我们通过分析SpringBoot中的源码来看他是如何实现的。

---

## 二、源码追踪看SpringBoot应用如何实现?

我们通过看Jconsole工具,可以看到工具里面的类名叫SpringApplication,目录是admin,于是我们就根据这个推测SpringBoot中的命名,果然我们找到两个实现类。

1. SpringApplicationAdminMXBean

这个类就是JMX中的MBean,我们可以简单理解这个里面的方法都是可以通过Jconsole来调用的。
通过将这个类注册给JMX管理器就能实现在Jconsole中的数据展示。

首先看SpringApplicationAdminMXBean

public interface SpringApplicationAdminMXBean {
   //是否可读
   boolean isReady();
   //是否web应用
   boolean isEmbeddedWebApplication();
   //获取配置信息
   String getProperty(String key);
   //下线应用
   void shutdown();
}

实现类SpringApplicationAdmin,是SpringApplicationAdminMXBeanRegistrar的内部类

private class SpringApplicationAdmin implements SpringApplicationAdminMXBean {
    // 是否可读,当应用还没起来时候这个值是false
      @Override
      public boolean isReady() {
         return SpringApplicationAdminMXBeanRegistrar.this.ready;
      }
      // 是否是web应用
      @Override
      public boolean isEmbeddedWebApplication() {
         return SpringApplicationAdminMXBeanRegistrar.this.embeddedWebApplication;
      }
    // 从Spring的配置信息中实时读取值
      @Override
      public String getProperty(String key) {
         return SpringApplicationAdminMXBeanRegistrar.this.environment.getProperty(key);
      }
    // 关闭Spring应用
      @Override
      public void shutdown() {
         logger.info("Application shutdown requested.");
         SpringApplicationAdminMXBeanRegistrar.this.applicationContext.close();
      }

   }

2. SpringApplicationAdminMXBeanRegistrar

提供注册能力。这个类中我们可以知道如何注册JMX以及如何取消注册。下面我看这个类如何利用Spring提供的接口能力,来实现应用下线。及注册到JMX上的吧。

自动化配置将SpringApplicationAdminMXBeanRegistrar声明成一个Spring中的Bean对象。并配置JMX中的命名及目录。

### 1. ApplicationContextAware

获得读取上下文能力。在Spring容器中一个bean如何实现了该方法则就可以获取上下文对象。

   @Override
   public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
      Assert.state(applicationContext instanceof ConfigurableApplicationContext,
            "ApplicationContext does not implement ConfigurableApplicationContext");
      this.applicationContext = (ConfigurableApplicationContext) applicationContext;
   }

### 2. GenericApplicationListener

获取处理事件的能力,同样在Spring中只要实现该接口,就获取了事件监听的能力,不过具体监听什么事件要自己去判断。大家可以根据例子
来理解。

  // 根据事件泛型判断是否需要处理,这里判断如果是ApplicationReadyEvent和WebServerInitializedEvent
  // 事件就处理
  @Override
   public boolean supportsEventType(ResolvableType eventType) {
      Class<?> type = eventType.getRawClass();
      if (type == null) {
         return false;
      }
      return ApplicationReadyEvent.class.isAssignableFrom(type)
            || WebServerInitializedEvent.class.isAssignableFrom(type);
   }

   @Override
   public boolean supportsSourceType(Class<?> sourceType) {
      return true;
   }

   @Override
   public void onApplicationEvent(ApplicationEvent event) {
    // 如果Spring已经准备好了,就将this.ready = true;
      if (event instanceof ApplicationReadyEvent) {
         onApplicationReadyEvent((ApplicationReadyEvent) event);
      }
    // 如果是Web应用,this.embeddedWebApplication = true
      if (event instanceof WebServerInitializedEvent) {
         onWebServerInitializedEvent((WebServerInitializedEvent) event);
      }
   }
   //优先级
   @Override
   public int getOrder() {
      return Ordered.HIGHEST_PRECEDENCE;
   }

   void onApplicationReadyEvent(ApplicationReadyEvent event) {
      if (this.applicationContext.equals(event.getApplicationContext())) {
         this.ready = true;
      }
   }

   void onWebServerInitializedEvent(WebServerInitializedEvent event) {
      if (this.applicationContext.equals(event.getApplicationContext())) {
         this.embeddedWebApplication = true;
      }
   }

### 3. EnvironmentAware

获取应用配置信息, 和上面一样实现了Aware结尾的接口,都能获取对象的Spring内容的对象实例,然后我们就可以根据该实例,来进行功能扩展。

@Override
   public void setEnvironment(Environment environment) {
      this.environment = environment;
   }

### 4. InitializingBean

这里就要着重看了,在初始化时候将MBean注册到JMX上。当然我们可以通过 @PostConstruct注解来声明初始化方法。

@Override
   public void afterPropertiesSet() throws Exception {
      MBeanServer server = ManagementFactory.getPlatformMBeanServer();
      server.registerMBean(new SpringApplicationAdmin(), this.objectName);
      if (logger.isDebugEnabled()) {
         logger.debug("Application Admin MBean registered with name '" + this.objectName + "'");
      }
   }

### 5. DisposableBean

应用销毁时候,取消注册。同样我们也可以用@PreDestroy注解来实现

@Override
   public void destroy() throws Exception {
      ManagementFactory.getPlatformMBeanServer().unregisterMBean(this.objectName);
   }

通过对SpringBoot应用源码的追踪,我们大概已经明白JMX的实际意义了,并且能自定义一个能提供类似能力的MBean了吧,但是JMX能做的远远不止如此。

## 三、自定义MBean

注意接口名必须是MBean结尾,实现类必须去掉MBean

如CustomMBean接口对应的实现类必须是Custom。

### 1. 代码实现

@Component
public class CustomMbeanRegistrar implements ApplicationContextAware, InitializingBean, DisposableBean {
    private ConfigurableApplicationContext applicationContext;
    private ObjectName objectName = new ObjectName("com.example.demo:type=CustomAdmin,name=CustomMXBean");

    public CustomMbeanRegistrar() throws MalformedObjectNameException {
    }
    @Override
    public void destroy() throws Exception {
        ManagementFactory.getPlatformMBeanServer().unregisterMBean(this.objectName);
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        MBeanServer server = ManagementFactory.getPlatformMBeanServer();
        server.registerMBean(new Custom(), this.objectName);
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = (ConfigurableApplicationContext) applicationContext;
    }

    public interface CustomMBean {
        int getDatabaseConnectionPoolSize();
        void customShutdown();
    }

    private class Custom implements CustomMBean {

        /**
         * 获取数据库连接池大小
         *
         * @return 模拟
         */
        @Override
        public int getDatabaseConnectionPoolSize() {
            return new Random().nextInt(100);
        }

        /**
         * 自定义一个销毁方法
         */
        public void customShutdown() {
            CustomMbeanRegistrar.this.applicationContext.close();
        }
    }
}

### 2. 演示

## 四、总结

通过前面的演示,大概我们对JMX在实际中的用处有一个大概的了解了吧。根据这个特性,我们就可以根据我们的需求来定制属于自己的能力。

最后求关注,求订阅,谢谢你的阅读!

相关文章
|
8月前
|
设计模式 前端开发 Java
使用Java MVC开发高效、可扩展的Web应用
本文将介绍如何使用Java MVC(Model-View-Controller)模式来开发高效、可扩展的Web应用。我们将深入探讨MVC模式的核心概念,以及如何在Java中实现这一体系结构。通过合理地分离业务逻辑、用户界面和数据处理,我们可以构建出易于维护和扩展的Web应用程序。
|
15天前
|
设计模式 Java API
Java 可扩展 API 设计:打造灵活的应用架构
【4月更文挑战第27天】设计可扩展的 API 是构建灵活、易于维护的应用程序架构的关键。Java 提供了丰富的工具和技术来实现这一目标,使开发者能够构建具有高度可扩展性的应用程序。
37 4
|
3天前
|
Java 编译器 开发者
Java一分钟之-继承:复用与扩展类的特性
【5月更文挑战第9天】本文探讨了Java中的继承机制,通过实例展示了如何使用`extends`创建子类继承父类的属性和方法。文章列举了常见问题和易错点,如构造器调用、方法覆盖、访问权限和类型转换,并提供了解决方案。建议深入理解继承原理,谨慎设计类结构,利用抽象类和接口以提高代码复用和扩展性。正确应用继承能构建更清晰、灵活的代码结构,提升面向对象设计能力。
9 0
|
18天前
|
设计模式 JavaScript Java
[设计模式Java实现附plantuml源码~结构型] 扩展系统功能——装饰模式
[设计模式Java实现附plantuml源码~结构型] 扩展系统功能——装饰模式
|
18天前
|
负载均衡 Java 数据库连接
Java从入门到精通:4.2.2学习新技术与框架——不断扩展自己的知识面,跟上技术的发展趋势
Java从入门到精通:4.2.2学习新技术与框架——不断扩展自己的知识面,跟上技术的发展趋势
|
2月前
|
Java API Maven
|
5月前
|
算法 Java
给定一个字符串数组,如何找到其中最长的回文子串? 要求:编写一个Java函数,输入一个字符串数组,输出其中最长的回文子串。要求时间复杂度为O(n^2)。可以考虑使用动态规划或中心扩展的方法来优化算法。
给定一个字符串数组,如何找到其中最长的回文子串? 要求:编写一个Java函数,输入一个字符串数组,输出其中最长的回文子串。要求时间复杂度为O(n^2)。可以考虑使用动态规划或中心扩展的方法来优化算法。
50 1
|
6月前
|
Java
Java接口:实现多重继承,促进代码复用与扩展的强大工具
Java接口:实现多重继承,促进代码复用与扩展的强大工具
|
7月前
|
设计模式 Java 数据库连接
JAVA设计模式8:装饰模式,动态地将责任附加到对象上,扩展对象的功能
JAVA设计模式8:装饰模式,动态地将责任附加到对象上,扩展对象的功能
|
8月前
|
Java
JAVA赋值和赋值扩展运算符
JAVA赋值和赋值扩展运算符
38 0