模仿Tomcat的BIO,NIO线程模型

简介:

模仿Tomcat的BIO模型,来一个消息,分配一个线程处理.
则主线程池代码如下
package com.guanjian;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**

  • Created by Administrator on 2018/7/10.
    */

public class ThreadPool {

private ExecutorService service;
private List<MessageTask> tasks;
private int fixedThreadNum = 0;
private List<String> messages;
private MessageHandler messageHandler;
public ThreadPool(int fixedThreadNum,List<String> messages,MessageHandler messageHandler) {
    this.fixedThreadNum = fixedThreadNum;
    this.messages = messages;
    this.messageHandler = messageHandler;
    service = Executors.newFixedThreadPool(fixedThreadNum);
    Runtime.getRuntime().addShutdownHook(new Thread() {
        public void run() {
            shutdownGracefully(service);
        }
    });
}
public void shutdownGracefully(ExecutorService ThreadPool) {
    ShutdownPool.shutdownThreadPool(ThreadPool, "main-pool");
}

public void startup() {
    tasks = new ArrayList<>();
    MessageTask messageTask = (fixedThreadNum == 0 ? new SequentialMessageTask(messageHandler,messages) : new ConcurrentMessageTask(messageHandler,messages));
    for (String message:messages) {
        tasks.add(messageTask);
        service.execute(messageTask);
    }
}

}
它是通过线程数fixedThreadNum来区分使用哪种线程模型.
package com.guanjian;

/**

  • Created by Administrator on 2018/7/10.
    */

public interface MessageHandler {

public void execute(String message);

}
package com.guanjian;

/**

  • Created by Administrator on 2018/7/10.
    */

public class MessageHandlerImpl implements MessageHandler {

@Override
public void execute(String message) {
    System.out.println(message);
}

}
以上是消息处理器的接口和实现类
package com.guanjian;

import java.util.List;

/**

  • Created by Administrator on 2018/7/10.
    */

public abstract class MessageTask implements Runnable {

protected MessageHandler messageHandler;
protected  List<String> messages;

MessageTask(MessageHandler messageHandler,List<String> messages) {
    this.messageHandler = messageHandler;
    this.messages = messages;
}
@Override
public void run() {
    for (String message:messages) {
        handlerMessage(message);
    }
}
protected abstract void handlerMessage(String message);

}
消息任务抽象类实现了Runnable线程接口,以不同的子类来实现BIO,NIO线程模型,具体在抽象方法handlerMessage中实现.
package com.guanjian;

import java.util.List;

/**

  • Created by Administrator on 2018/7/10.
    */

public class SequentialMessageTask extends MessageTask {

SequentialMessageTask(MessageHandler messageHandler, List<String> messages) {
    super(messageHandler, messages);
}

@Override
protected void handlerMessage(String message) {
    messageHandler.execute(message);
}

}
BIO线程模型子类,通过主线程池来分配线程处理.
package com.guanjian;

import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**

  • Created by Administrator on 2018/7/10.
    */

public class ConcurrentMessageTask extends MessageTask {

private ExecutorService asyncService = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors() * 2);
ConcurrentMessageTask(MessageHandler messageHandler, List<String> messages) {
    super(messageHandler, messages);
}

@Override
protected void handlerMessage(String message) {
    asyncService.submit(new Runnable() {
        @Override
        public void run() {
            messageHandler.execute(message);
        }
    });
}
protected void shutdown() {
    ShutdownPool.shutdownThreadPool(asyncService,"async-pool-" + Thread.currentThread().getId());
}

}
NIO线程模型,不再使用主线程池来分配线程,而是异步线程池,类比于Netty中的Worker线程池,从BOSS线程池中接管消息处理.
package com.guanjian;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.TimeUnit;

/**

  • Created by Administrator on 2018/7/10.
    */

public class ShutdownPool {

private static Logger log = LoggerFactory.getLogger(ThreadPool.class);
/**
 * 优雅关闭线程池
 * @param threadPool
 * @param alias
 */
public static void shutdownThreadPool(ExecutorService threadPool, String alias) {
    log.info("Start to shutdown the thead pool: {}", alias);

    threadPool.shutdown(); // 使新任务无法提交.
    try {
        // 等待未完成任务结束
        if (!threadPool.awaitTermination(60, TimeUnit.SECONDS)) {
            threadPool.shutdownNow(); // 取消当前执行的任务
            log.warn("Interrupt the worker, which may cause some task inconsistent. Please check the biz logs.");

            // 等待任务取消的响应
            if (!threadPool.awaitTermination(60, TimeUnit.SECONDS))
                log.error("Thread pool can't be shutdown even with interrupting worker threads, which may cause some task inconsistent. Please check the biz logs.");
        }
    } catch (InterruptedException ie) {
        // 重新取消当前线程进行中断
        threadPool.shutdownNow();
        log.error("The current server thread is interrupted when it is trying to stop the worker threads. This may leave an inconcistent state. Please check the biz logs.");

        // 保留中断状态
        Thread.currentThread().interrupt();
    }

    log.info("Finally shutdown the thead pool: {}", alias);
}

}
最后是线程池的优雅关闭,无论是主线程池还是异步线程池皆调用该方法实现优雅关闭.

以上只是模型代码,具体可替换成具体需要的业务代码来达到业务性能的提升.

相关文章
|
3月前
|
网络协议 Dubbo Java
一文搞懂NIO、AIO、BIO的核心区别(建议收藏)
本文详细解析了NIO、AIO、BIO的核心区别,NIO的三个核心概念,以及NIO在Java框架中的应用等。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
一文搞懂NIO、AIO、BIO的核心区别(建议收藏)
|
3天前
|
存储 缓存 关系型数据库
MySQL底层概述—3.InnoDB线程模型
InnoDB存储引擎采用多线程模型,包含多个后台线程以处理不同任务。主要线程包括:IO Thread负责读写数据页和日志;Purge Thread回收已提交事务的undo日志;Page Cleaner Thread刷新脏页并清理redo日志;Master Thread调度其他线程,定时刷新脏页、回收undo日志、写入redo日志和合并写缓冲。各线程协同工作,确保数据一致性和高效性能。
MySQL底层概述—3.InnoDB线程模型
|
3月前
|
Java
BIO、NIO、AIO 有什么区别
BIO(阻塞I/O)模型中,服务器实现模式为一个连接一个线程;NIO(非阻塞I/O)使用单线程或少量线程处理多个请求;AIO(异步I/O)则是在NIO基础上进一步优化,采用事件通知机制,提高并发处理能力。
99 6
|
3月前
|
消息中间件 监控 Java
BIO、NIO、AIO在不同场景下的应用对比
BIO(阻塞I/O)、NIO(非阻塞I/O)和AIO(异步I/O)是Java中处理I/O操作的三种模式。BIO适用于连接数少且稳定的场景;NIO通过非阻塞模式提高并发处理能力,适合高并发场景;AIO则完全异步,适合需要高效、低延迟的I/O操作场景。
248 4
|
4月前
|
并行计算 JavaScript 前端开发
单线程模型
【10月更文挑战第15天】
|
4月前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
43 1
|
4月前
|
安全 调度 C#
STA模型、同步上下文和多线程、异步调度
【10月更文挑战第19天】本文介绍了 STA 模型、同步上下文和多线程、异步调度的概念及其优缺点。STA 模型适用于单线程环境,确保资源访问的顺序性;同步上下文和多线程提高了程序的并发性和响应性,但增加了复杂性;异步调度提升了程序的响应性和资源利用率,但也带来了编程复杂性和错误处理的挑战。选择合适的模型需根据具体应用场景和需求进行权衡。
|
4月前
|
Java Linux 应用服务中间件
【编程进阶知识】高并发场景下Bio与Nio的比较及原理示意图
本文介绍了在Linux系统上使用Tomcat部署Java应用程序时,BIO(阻塞I/O)和NIO(非阻塞I/O)在网络编程中的实现和性能差异。BIO采用传统的线程模型,每个连接请求都会创建一个新线程进行处理,导致在高并发场景下存在严重的性能瓶颈,如阻塞等待和线程创建开销大等问题。而NIO则通过事件驱动机制,利用事件注册、事件轮询器和事件通知,实现了更高效的连接管理和数据传输,避免了阻塞和多级数据复制,显著提升了系统的并发处理能力。
107 0
|
7月前
|
Java 大数据
解析Java中的NIO与传统IO的区别与应用
解析Java中的NIO与传统IO的区别与应用
|
7月前
|
Java
Java中的NIO编程详解
Java中的NIO编程详解