服务器监控功能(3种方案)

2023-11-03

一、Actuator监控

Actuator是Springboot提供的用来对应用系统进行自省和监控的功能模块,借助于Actuator开发者可以很方便地对应用系统某些监控指标进行查看、统计等。

部署简单、直接调接口拿值、数据较分散,需处理

1. 添加依赖
<!-- 引入Actuator监控依赖 -->
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
2. application.yaml配置
#actuator监控配置
 management:
 	endpoints:
 		web:
   			exposure:
     			#默认值访问health,info端点  用*可以包含全部端点
     			include: '*'
     		#配置路径
        	base-path: /system/actuator
  	endpoint:
    	health:
      		#获得健康检查中所有指标的详细信息
      		show-details: always
3. 启动项目,访问

以下显示的路径,都是可以被健康检查的指标

其中谷歌浏览器显示格式为 JSON,是因为下载了 JSON-handle 插件

  • 访问路径:项目路径+actuator配置路径

  • Actuator健康项:

二、SpringBoot Admin(单体)

可监控的信息包含:应用状态、内存、线程、堆栈等等,比较全面的监控了 Spring Boot 应用的整个生命周期。

在Spring Boot Actuator的基础上提供简洁的可视化WEB UI

SpringBoot Admin 分为服务端(spring-boot-admin-server)和客户端(spring-boot-admin-client),客户端就是指需要被监控的应用端(项目启动位置)

1. Admin Server 端
  • 新建一个项目模块,项目依赖

    <dependency>
        <groupId>de.codecentric</groupId>
        <artifactId>spring-boot-admin-starter-server</artifactId>
        <version>2.1.0</version>
     </dependency>
     <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
     </dependency>
    
  • application.yaml 配置文件

    # 配置tomcat访问端口
    server:
      port: 8000
    
  • 启动类

    import de.codecentric.boot.admin.server.config.EnableAdminServer;
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
    import org.springframework.context.annotation.Configuration;
    
    /**
     * Springboot admin 服务器监控 -- 服务端
     *
     * @Author: changge
     * @date 2021/8/25 17:43
     * @return null
     **/
    @Configuration
    @EnableAutoConfiguration
    @EnableAdminServer
    public class AdminServerApplication {
    
        public static void main(String[] args) {
            SpringApplication.run(AdminServerApplication.class, args);
        }
    }
    
  • 访问浏览器 http://localhost:8000

2. Admin Client 端
  • 在原本项目基础上,添加依赖

    <dependencies>
        <dependency>
          <groupId>de.codecentric</groupId>
          <artifactId>spring-boot-admin-starter-client</artifactId>
          <version>2.1.0</version>
        </dependency>
        <dependency>
          <groupId>org.springframework.boot</groupId>
          <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
    </dependencies>
    
  • 配置文件

    # 配置访问端口
    server:
      port: 20211
    spring:
      application:
        name: Admin Client
      # 配置AdminServer的地址
      boot:
        admin:
          client:
            url: http://localhost:8000
    # 打开客户端的监控
    management:
      endpoints:
        web:
          exposure:
            include: *
    
  • 客户端启动类

    @SpringBootApplication
    public class AdminClientApplication {
      public static void main(String[] args) {
        SpringApplication.run(AdminClientApplication.class, args);
      }
    }
    
  • 先启动服务端,再启动客户端

    启动 Client 端,Admin 服务端会自动检查到客户端的变化,并展示其应用

    页面会展示被监控的服务列表,点击详项目名称会进入此应用的详细监控信息

三、OSHI

基于JNA的免费的本地操作系统和Java的硬件信息库,可以跨平台,获取监控信息简单

1. pom.xml依赖
<!-- 获取系统信息 -->
<dependency>
	<groupId>com.github.oshi</groupId>
     <artifactId>oshi-core</artifactId>
     <version>5.6.0</version>
 </dependency>
2. 接口返回数据
3. 代码
  • Controller

    import com.lzby.tqj.dto.AjaxResult;
    import com.lzby.tqj.entity.Server;
    import io.swagger.*;
    import org.springframework.web.bind.annotation.*;
    
    /**
     * 服务器监控
     **/
    @CrossOrigin(origins = "*", maxAge = 3600)
    @Api(tags = "系统-服务器监控")
    @RestController
    @RequestMapping("/monitor/server")
    public class ServerController {
    
        @ApiOperation(value = "获得服务器相关信息", notes = "获得服务器相关信息")
        @PostMapping(value = "/getInfo")
        public AjaxResult getInfo() throws Exception {
            Server server = new Server();
            server.copyTo();
            return AjaxResult.success(server);
        }
    }
    
  • entity - Server

    package com.lzby.tqj.entity;
    
    import com.lzby.tqj.common.IpUtils;
    import com.lzby.tqj.common.util.Arith;
    import com.lzby.tqj.entity.server.*;
    import oshi.*;
    
    import java.net.InetAddress;
    
    import java.util.*;
    
    /**
     * 服务器相关信息
     **/
    public class Server {
        private static final int OSHI_WAIT_SECOND = 1000;
    
        // CPU相关信息
        private Cpu cpu = new Cpu();
    
        // 內存相关信息
        private Mem mem = new Mem();
    
        // JVM相关信息
        private Jvm jvm = new Jvm();
    
        // 服务器相关信息
        private Sys sys = new Sys();
    
        // 磁盘相关信息
        private List<SysFile> sysFiles = new LinkedList<SysFile>();
    
        public Cpu getCpu() {
            return cpu;
        }
    
        public void setCpu(Cpu cpu) {
            this.cpu = cpu;
        }
    
        public Mem getMem() {
            return mem;
        }
    
        public void setMem(Mem mem) {
            this.mem = mem;
        }
    
        public Jvm getJvm() {
            return jvm;
        }
    
        public void setJvm(Jvm jvm) {
            this.jvm = jvm;
        }
    
        public Sys getSys() {
            return sys;
        }
    
        public void setSys(Sys sys) {
            this.sys = sys;
        }
    
        public List<SysFile> getSysFiles() {
            return sysFiles;
        }
    
        public void setSysFiles(List<SysFile> sysFiles) {
            this.sysFiles = sysFiles;
        }
    
        public void copyTo() throws Exception {
            SystemInfo si = new SystemInfo();
            HardwareAbstractionLayer hal = si.getHardware();
            setCpuInfo(hal.getProcessor());
            setMemInfo(hal.getMemory());
            setSysInfo();
            setJvmInfo();
            setSysFiles(si.getOperatingSystem());
        }
    
        /**
         * 设置CPU信息
         */
        private void setCpuInfo(CentralProcessor processor) {
            // CPU信息
            long[] prevTicks = processor.getSystemCpuLoadTicks();
            Util.sleep(OSHI_WAIT_SECOND);
            long[] ticks = processor.getSystemCpuLoadTicks();
            long nice = ticks[TickType.NICE.getIndex()] - prevTicks[TickType.NICE.getIndex()];
            long irq = ticks[TickType.IRQ.getIndex()] - prevTicks[TickType.IRQ.getIndex()];
            long softirq = ticks[TickType.SOFTIRQ.getIndex()] - prevTicks[TickType.SOFTIRQ.getIndex()];
            long steal = ticks[TickType.STEAL.getIndex()] - prevTicks[TickType.STEAL.getIndex()];
            long cSys = ticks[TickType.SYSTEM.getIndex()] - prevTicks[TickType.SYSTEM.getIndex()];
            long user = ticks[TickType.USER.getIndex()] - prevTicks[TickType.USER.getIndex()];
            long iowait = ticks[TickType.IOWAIT.getIndex()] - prevTicks[TickType.IOWAIT.getIndex()];
            long idle = ticks[TickType.IDLE.getIndex()] - prevTicks[TickType.IDLE.getIndex()];
            long totalCpu = user + nice + cSys + idle + iowait + irq + softirq + steal;
            cpu.setCpuNum(processor.getLogicalProcessorCount());
            cpu.setTotal(totalCpu);
            cpu.setSys(cSys);
            cpu.setUsed(user);
            cpu.setWait(iowait);
            cpu.setFree(idle);
        }
    
        /**
         * 设置内存信息
         */
        private void setMemInfo(GlobalMemory memory) {
            mem.setTotal(memory.getTotal());
            mem.setUsed(memory.getTotal() - memory.getAvailable());
            mem.setFree(memory.getAvailable());
        }
    
        /**
         * 设置服务器信息
         */
        private void setSysInfo() {
            Properties props = System.getProperties();
            sys.setComputerName(getHostName());
            sys.setComputerIp(getHostIp());
            sys.setOsName(props.getProperty("os.name"));
            sys.setOsArch(props.getProperty("os.arch"));
            sys.setUserDir(props.getProperty("user.dir"));
        }
    
        /**
         * 设置Java虚拟机
         */
        private void setJvmInfo() throws UnknownHostException {
            Properties props = System.getProperties();
            jvm.setTotal(Runtime.getRuntime().totalMemory());
            jvm.setMax(Runtime.getRuntime().maxMemory());
            jvm.setFree(Runtime.getRuntime().freeMemory());
            jvm.setVersion(props.getProperty("java.version"));
            jvm.setHome(props.getProperty("java.home"));
        }
    
        /**
         * 设置磁盘信息
         */
        private void setSysFiles(OperatingSystem os) {
            FileSystem fileSystem = os.getFileSystem();
            List<OSFileStore> fsArray = fileSystem.getFileStores();
            for (OSFileStore fs : fsArray) {
                long free = fs.getUsableSpace();
                long total = fs.getTotalSpace();
                long used = total - free;
                SysFile sysFile = new SysFile();
                sysFile.setDirName(fs.getMount());
                sysFile.setSysTypeName(fs.getType());
                sysFile.setTypeName(fs.getName());
                sysFile.setTotal(convertFileSize(total));
                sysFile.setFree(convertFileSize(free));
                sysFile.setUsed(convertFileSize(used));
                sysFile.setUsage(Arith.mul(Arith.div(used, total, 4), 100));
                sysFiles.add(sysFile);
            }
        }
    
        /**
         * 字节转换
         *
         * @param size 字节大小
         * @return 转换后值
         */
        public String convertFileSize(long size) {
            long kb = 1024;
            long mb = kb * 1024;
            long gb = mb * 1024;
            if (size >= gb) {
                return String.format("%.1f GB", (float) size / gb);
            } else if (size >= mb) {
                float f = (float) size / mb;
                return String.format(f > 100 ? "%.0f MB" : "%.1f MB", f);
            } else if (size >= kb) {
                float f = (float) size / kb;
                return String.format(f > 100 ? "%.0f KB" : "%.1f KB", f);
            } else {
                return String.format("%d B", size);
            }
        }
        
        public static String getHostIp() {
            try {
                return InetAddress.getLocalHost().getHostAddress();
            } catch (UnknownHostException e) {
            }
            return "127.0.0.1";
        }
    
        public static String getHostName() {
            try {
                return InetAddress.getLocalHost().getHostName();
            } catch (UnknownHostException e) {
            }
            return "未知";
        }
    }
    
    
  • entity - Cpu

    package com.lzby.tqj.entity.server;
    
    import com.lzby.tqj.common.util.Arith;
    
    /**
     * CPU相关信息
     */
    public class Cpu
    {
        // 核心数
        private int cpuNum;
    
        // CPU总的使用率
        private double total;
    
        // CPU系统使用率
        private double sys;
    
        // CPU用户使用率
        private double used;
    
        // CPU当前等待率
        private double wait;
    
        // CPU当前空闲率
        private double free;
    
        public int getCpuNum(){
            return cpuNum;
        }
    
        public void setCpuNum(int cpuNum){
            this.cpuNum = cpuNum;
        }
    
        public double getTotal(){
            return Arith.round(Arith.mul(total, 100), 2);
        }
    
        public void setTotal(double total){
            this.total = total;
        }
    
        public double getSys(){
            return Arith.round(Arith.mul(sys / total, 100), 2);
        }
    
        public void setSys(double sys){
            this.sys = sys;
        }
    
        public double getUsed(){
            return Arith.round(Arith.mul(used / total, 100), 2);
        }
    
        public void setUsed(double used){
            this.used = used;
        }
    
        public double getWait(){
            return Arith.round(Arith.mul(wait / total, 100), 2);
        }
    
        public void setWait(double wait){
            this.wait = wait;
        }
    
        public double getFree(){
            return Arith.round(Arith.mul(free / total, 100), 2);
        }
    
        public void setFree(double free){
            this.free = free;
        }
    }
    
  • entity - Mem

    package com.lzby.tqj.entity.server;
    
    import com.lzby.tqj.common.util.Arith;
    
    /**
     * 內存相关信息
     */
    public class Mem{
        // 内存总量
        private double total;
    
        // 已用内存
        private double used;
    
        // 剩余内存
        private double free;
    
        public double getTotal(){
            return Arith.div(total, (1024 * 1024 * 1024), 2);
        }
    
        public void setTotal(long total){
            this.total = total;
        }
    
        public double getUsed(){
            return Arith.div(used, (1024 * 1024 * 1024), 2);
        }
    
        public void setUsed(long used){
            this.used = used;
        }
    
        public double getFree(){
            return Arith.div(free, (1024 * 1024 * 1024), 2);
        }
    
        public void setFree(long free){
            this.free = free;
        }
    
        public double getUsage(){
            return Arith.mul(Arith.div(used, total, 4), 100);
        }
    }
    
  • entity - Jvm

    package com.lzby.tqj.entity.server;
    
    import java.lang.management.ManagementFactory;
    import com.lzby.tqj.common.util.Arith;
    import com.lzby.tqj.common.util.DateUtils;
    
    /**
     * JVM相关信息
     **/
    public class Jvm{
        // 当前JVM占用的内存总数(M)
        private double total;
    
        // JVM最大可用内存总数(M)
        private double max;
    
        // JVM空闲内存(M)
        private double free;
    
        // JDK版本
        private String version;
    
        // JDK路径
        private String home;
    
        public double getTotal(){
            return Arith.div(total, (1024 * 1024), 2);
        }
    
        public void setTotal(double total){
            this.total = total;
        }
    
        public double getMax(){
            return Arith.div(max, (1024 * 1024), 2);
        }
    
        public void setMax(double max){
            this.max = max;
        }
    
        public double getFree(){
            return Arith.div(free, (1024 * 1024), 2);
        }
    
        public void setFree(double free){
            this.free = free;
        }
    
        public double getUsed(){
            return Arith.div(total - free, (1024 * 1024), 2);
        }
    
        public double getUsage(){
            return Arith.mul(Arith.div(total - free, total, 4), 100);
        }
    
        // 获取JDK名称
        public String getName(){
            return ManagementFactory.getRuntimeMXBean().getVmName();
        }
    
        public String getVersion(){
            return version;
        }
    
        public void setVersion(String version){
            this.version = version;
        }
    
        public String getHome(){
            return home;
        }
    
        public void setHome(String home){
            this.home = home;
        }
    
        // JDK启动时间
        public String getStartTime(){
            return DateUtils.parseDateToStr(DateUtils.YYYY_MM_DD_HH_MM_SS, DateUtils.getServerStartDate());
        }
    
        // JDK运行时间
        public String getRunTime()
        {
            return DateUtils.getDatePoor(DateUtils.getNowDate(), DateUtils.getServerStartDate());
        }
    }
    
  • entity - Jvm

    package com.lzby.tqj.entity.server;
    
    import lombok.Data;
    
    /**
     * 系统相关信息
     */
    @Data
    public class Sys{
        
        // 服务器名称
        private String computerName;
    
        // 服务器Ip
        private String computerIp;
    
        // 项目路径
        private String userDir;
    
        // 操作系统
        private String osName;
    
        // 系统架构
        private String osArch;
    }
    
  • 工具类 Arith

    package com.lzby.tqj.common.util;
    
    import java.math.BigDecimal;
    import java.math.RoundingMode;
    
    /**
     * 精确的浮点数运算
     **/
    public class Arith{
    
        /** 默认除法运算精度 */
        private static final int DEF_DIV_SCALE = 10;
    
        /** 这个类不能实例化 */
        private Arith(){}
    
        /**
         * 提供精确的加法运算。
         * @param v1 被加数
         * @param v2 加数
         * @return 两个参数的和
         */
        public static double add(double v1, double v2){
            BigDecimal b1 = new BigDecimal(Double.toString(v1));
            BigDecimal b2 = new BigDecimal(Double.toString(v2));
            return b1.add(b2).doubleValue();
        }
    
        /**
         * 提供精确的乘法运算。
         * @param v1 被乘数
         * @param v2 乘数
         * @return 两个参数的积
         */
        public static double mul(double v1, double v2){
            BigDecimal b1 = new BigDecimal(Double.toString(v1));
            BigDecimal b2 = new BigDecimal(Double.toString(v2));
            return b1.multiply(b2).doubleValue();
        }
    
        /**
         * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指
         * 定精度,以后的数字四舍五入。
         * @param v1 被除数
         * @param v2 除数
         * @param scale 表示表示需要精确到小数点以后几位。
         * @return 两个参数的商
         */
        public static double div(double v1, double v2, int scale){
            if (scale < 0){
                throw new IllegalArgumentException(
                        "The scale must be a positive integer or zero");
            }
            BigDecimal b1 = new BigDecimal(Double.toString(v1));
            BigDecimal b2 = new BigDecimal(Double.toString(v2));
            if (b1.compareTo(BigDecimal.ZERO) == 0){
                return BigDecimal.ZERO.doubleValue();
            }
            return b1.divide(b2, scale, RoundingMode.HALF_UP).doubleValue();
        }
    
        /**
         * 提供精确的小数位四舍五入处理。
         * @param v 需要四舍五入的数字
         * @param scale 小数点后保留几位
         * @return 四舍五入后的结果
         */
        public static double round(double v, int scale){
            if (scale < 0){
                throw new IllegalArgumentException(
                        "The scale must be a positive integer or zero");
            }
            BigDecimal b = new BigDecimal(Double.toString(v));
            BigDecimal one = new BigDecimal("1");
            return b.divide(one, scale, RoundingMode.HALF_UP).doubleValue();
        }
    }
    
    
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

服务器监控功能(3种方案) 的相关文章

随机推荐

  • sklearn函数汇总

    Python笔记 sklearn函数汇总 Summer Memories 关注他 28 人 赞同了该文章 1 拆分数据集为训练集和测试集 from sklearn model selection import train test spli
  • Python自动化测试框架有哪些?

    常用的Python自动化测试框架有Robot Framework Pytest UnitTest PyUnit Behave Lettuce 软件测试的自动化在预设条件下运行系统或应用程序 评估运行结果 预先条件应包括正常条件和异常条件 自
  • gpu cpu 共享内存 提高传输速度_AMD扩展Infinity Fabric总线技术 CPU与GPU之间终于共享内存了...

    在AMD的锐龙处理器架构中 Infinity Fabric 以下简称IF 总线是个核心技术 有了它才可以让众多CCX模块互联互通 之前IF总线主要用于CPU核心之间连接 现在AMD终于开始用于EPYC CPU与Radeon显卡了 在OGHP
  • windows进程间通信

    摘 要 随着人们对应用程序的要求越来越高 单进程应用在许多场合已不能满足人们的要求 编写多进程 多线程程序成为现代程序设计的一个重要特点 在多进程程序设计中 进程间的通信是不可避免的 Microsoft Win32 API提供了多种进程间通
  • 引发的异常:“System.BadImageFormatException”(位于 halcondotnet.dll 中)

    System BadImageFormatException 类型的未经处理的异常在 halcondotnet dll 中发生 问题描述 解决办法 问题描述 报错信息如下 System BadImageFormatException 试图加
  • 一篇专题让你秒懂GCD死锁问题!

    一篇专题让你秒懂GCD死锁问题 阐述 1 什么是GCD GCD 全称 Grand Central Dispatch 可翻译为 牛逼的中枢调度器 它是纯C语言的 提供了非常多强大的函数 Grand是宏伟的 极重要的意思 GCD是提供了功能强大
  • 一个VSCode便可以免费画ER图

    为了找到一款好用的er图工具 我基本上把能用的途径都看了个遍 最后 好家伙不是收费就是要下载各式各样 大小不一的软件 下也就算了 还要收费 不收费就限制节点啊什么的 反正搞得是那个上火啊 为了帮助大家闭坑 特意写此博客 其实VScode就可
  • Buffer类的详解(转)

    这篇蚊帐转自 http zachary guo iteye com blog 1457542 作者写的非常好 是我看到的写nio最好的几篇蚊帐 但原文中有一些错误 还有我自己对这方面的一些理解 在这里一并更改了 Buffer 类是 java
  • 2023年交通与智慧城市国际会议(ICoTSC 2023)

    2023年交通与智慧城市国际会议 ICoTSC 2023 重要信息 会议网址 www icotsc org 会议时间 2023年7月28 30日 召开地点 长沙 截稿时间 2023年6月15日 录用通知 投稿后2周内 收录检索 EI Sco
  • ERROR: Failed building wheel for mayavi

    conda虚拟环境中安装mayavi 提示错误 Numpy is required to build Mayavi correctly please install it first ERROR Failed building wheel
  • Java Web快速开发(2~4)Java 基本程序结构、类、继承和接口

    文章目录 二 Java基本程序结构 1 数据类型 2 运算符 3 循环语句 三 类 四 继承和接口 1 多态 2 抽象类 3 接口定义 4 接口增强 5 clone 实现的只是浅拷贝 二 Java基本程序结构 1 数据类型 2 运算符 3
  • Android 全局手势识别原理

    本文是对全局手势识别进行分析 那什么是全局手势呢 简单来说就是在任何界面都需要识别的手势 比如 在任何界面从手机屏幕左侧滑动 当前的界面会退出 类似 back 键 我们知道 在 Android 系统中一个 Activity 在显示时 当对屏
  • 关于程序员35岁的坎:年龄不是挡板,当你匹配了这个年纪该有的能力还有什么畏惧!

    前言 关于程序员干到35岁 会不会失业或者是说35岁的坎 这里我分享一篇不错的文章 希望能给大家带来一些帮助和思考 本文可能可以适用到其他端 不过由于自己是一名 Android 开发工程师 所以就姑且从一名普通的 Android 开发工程师
  • 使用jsp+servlet实现留言板(学习使用)

    1 前端jsp页面 1 1前端总体代码
  • 数据治理体系完整指南(值得收藏)

    01 数据治理之数据资产管理 数据资产与数据资产管理的定义 随着大数据时代的到来 人们已经认识到数据是一种无形的宝贵资产 谷歌 Facebook 阿里巴巴 腾讯等企业市值高达数千亿美元 不仅在于其独特的商业模式和市场垄断地位 更多的估值是给
  • 虚拟服务器转移到实体服务器,虚拟主机转移到云服务器

    虚拟主机转移到云服务器 内容精选 换一换 场景描述绑定到云服务器上 实现云服务器连接公网的目的 搭配服务弹性云服务器ECS或裸金属服务器 虚拟私有云绑定云服务器实例场景描述NAT网关通过与弹性公网IP绑定 可以使多个云主机 弹性云服务器 裸
  • APP内存泄漏导致崩溃测试实践

    内存泄漏崩溃原因 由于程序运行后没有及时回收无用对象 导致系统没有足够内存分配新对象 会引起APP崩溃 Android的GC机制 Android GC机制沿用了java的GC机制 当需要新内存去分配对象的时候而剩余不够的时候 会触发GC 把
  • for循环中,外循环1000、内循环100和外循环100、内循环1000哪个性能更高?

    1 直接上代码 public class Demo public static void main String args System out println firstTime Demo first 毫秒 255毫秒 System ou
  • laravel+element-ui+vue 搭建详细教程

    1 可用composer安装laravel框架项目 composer create project laravel laravel myblog 安装完成进入项目 cd myblog 2 安装前端资源和编译 npm install npm
  • 服务器监控功能(3种方案)

    服务器监控功能 3种方案 一 Actuator监控 1 添加依赖 2 application yaml配置 3 启动项目 访问 二 SpringBoot Admin 单体 1 Admin Server 端 2 Admin Client 端