Spring boot tomcat使用自定义线程池监控线程数量告警

Spring boot tocmat 使用自定义线程池

线程池 接近最大线程数量 警戒值告警

修改tomcat线程池中线程名字

配置文件上代码

server:
  port: 9898
  servlet:
    context-path: /test
  tomcat:
    connection-timeout: 5000
    max-connections: 5
    accept-count: 5
tomcat_thread_max_number_warn: 350
tomcat_thread_max_number: 500
tomcat_waiter_number: 500

自定义线程池TomcatMonitorConfig

package com.company.config;

import lombok.extern.slf4j.Slf4j;
import org.apache.coyote.AbstractProtocol;
import org.apache.coyote.ProtocolHandler;
import org.apache.tomcat.util.security.PrivilegedSetTccl;
import org.apache.tomcat.util.threads.*;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory;
import org.springframework.boot.web.server.WebServerFactoryCustomizer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.concurrent.Executor;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;

@Slf4j
@Configuration
public class TomcatMonitorConfig {

    @Value("${tomcat_thread_max_number_warn}")
    private int maxThreadNumberWarn;

    @Value("${tomcat_thread_max_number}")
    private int maxThreadNumber;

    @Value("${tomcat_waiter_number}")
    private int waiterNumber;

    @Bean
    public WebServerFactoryCustomizer myWebServerFactoryCustomizer() {
        TaskThreadFactory tf = new TaskThreadFactory("tomcat-http" + "-exec-", true, 1);
        return new WebServerFactoryCustomizer<TomcatServletWebServerFactory>() {
            @Override
            public void customize(TomcatServletWebServerFactory factory) {
                Executor executor = (Executor)new ThreadPoolExecutor(Runtime.getRuntime().availableProcessors(),
                        maxThreadNumber,
                        60,
                        TimeUnit.SECONDS,
                        new LinkedBlockingQueue<>(waiterNumber),
                        tf
                );
                factory.addConnectorCustomizers((connector) -> {
                    ProtocolHandler handler = connector.getProtocolHandler();
                    TaskQueue taskqueue = new TaskQueue();
                    if (handler instanceof AbstractProtocol) {
                        AbstractProtocol<?> protocol = (AbstractProtocol<?>) handler;
                        protocol.setExecutor(executor);
                    }
                });
            }
        };
    }

    class TaskThreadFactory implements ThreadFactory {
        private final ThreadGroup group;
        private final AtomicInteger threadNumber = new AtomicInteger(1);
        private final String namePrefix;
        private final boolean daemon;
        private final int threadPriority;

        public TaskThreadFactory(String namePrefix, boolean daemon, int priority) {
            SecurityManager s = System.getSecurityManager();
            group = (s != null) ? s.getThreadGroup() : Thread.currentThread().getThreadGroup();
            this.namePrefix = namePrefix;
            this.daemon = daemon;
            this.threadPriority = priority;
        }

        @Override
        public Thread newThread(Runnable r) {
            int number = threadNumber.getAndIncrement();
            if(number>=maxThreadNumberWarn) {
                log.warn("Thread number is greater than "+maxThreadNumberWarn);
            }
            TaskThread t = new TaskThread(group, r, namePrefix + number);
            t.setDaemon(daemon);
            t.setPriority(threadPriority);

            // Set the context class loader of newly created threads to be the class
            // loader that loaded this factory. This avoids retaining references to
            // web application class loaders and similar.
            if (Constants.IS_SECURITY_ENABLED) {
                PrivilegedAction<Void> pa = new PrivilegedSetTccl(
                        t, getClass().getClassLoader());
                AccessController.doPrivileged(pa);
            } else {
                t.setContextClassLoader(getClass().getClassLoader());
            }

            return t;
        }
    }


}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值