自定义线程池:简单demo(+自定义ThreadFactory)实现

  • 线程池作用简述
  • 一、上demo(自定义线程池)
  • 二、ThreadFactory源码简述
  • 1、方法定义
  • 2、接口定义


线程池作用简述

线程池主要作用是为了线程复用,避免线程频繁创建。
线程池里的线程是 ThreadFactory帮助创建的。

一、上demo(自定义线程池)

可以跟踪线程池究竟在何时创建了多少线程,也可以自定义线程名称,组,优先级,甚至直接设定所有线程为守护线程。可以通过自定义线程池更加自由的设置池子里所有线程的状态。下面案例一方面记录线程的创建,另一方面把所有线程都设置为守护线程,当主线程退出后,将会强制销毁线程池,省了我手动关闭线程池了,哈哈

package com.test.threadPoolStu;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import java.util.concurrent.*;

/**自定义线程池*/
public class CustomizeThreadPool {
    public static void main(String[] args) throws InterruptedException {
        //自定义线程工厂 方式一:线程名称揉入业务含义
        // private static ThreadFactory customThreadFactory = new ThreadFactoryBuilder().setNameFormat("transAcount-pool-%d").build();

        //自定义线程工厂 方式二
        ThreadFactory customThreadFactory2=new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                Thread t=new Thread(r ); // Thread t=new Thread(r,"transAcount-pool-%d");
                t.setName("transAcount-pool-"+t.getId());
                t.setDaemon(true);//设置为守护线程
                System.out.println("create "+t.getId()+": tName="+t.getName());
                return t;
            }
        };

        MyTask task=new MyTask();
        ExecutorService es=new ThreadPoolExecutor(5,5,0L, TimeUnit.MILLISECONDS,
                new SynchronousQueue<Runnable>(),customThreadFactory2);

        for (int i=0;i<5;i++) {
            es.submit(task);
        }
        Thread.sleep(2000);
    }
    public static class MyTask implements Runnable{
        @Override
        public void run(){
            System.out.println(System.currentTimeMillis()+": Thread ID :"+Thread.currentThread().getId());
            try {
                Thread.sleep(100);
            }catch (InterruptedException e){
                System.out.println(e);
            }
        }
    }
}

结果:能获取到线程名称,以及什么时间创建的,而且任务轮训完线程池关闭。

com.test.threadPoolStu.CustomizeThreadPool
create 11: tName=transAcount-pool-11
create 12: tName=transAcount-pool-12
create 13: tName=transAcount-pool-13
1597155474887: Thread ID :11
1597155474888: Thread ID :12
create 14: tName=transAcount-pool-14
1597155474889: Thread ID :13
create 15: tName=transAcount-pool-15
1597155474889: Thread ID :14
1597155474889: Thread ID :15

Process finished with exit code 0

自定义线程池 看到这就可以了,但如果你想了解ThreadFactory的源码,可以继续

二、ThreadFactory源码简述

ThreadFactory就一很朴素的接口,只有一个方法newThread
看源码能知道,具体可以参看下面的线程工厂源码解读

1、方法定义

1、怎么使用?

①想用就实现这个接口,重写创建线程的方法;

②或者你可以通过接口回调的形式重写方法体。

使用的话,是放在ThreadPoolExecutor的构造里作为一参数。

java 线程池中线程ID的编号 java线程池threadfactory_线程池


2、源码的方法定义

方法定义上,至少咱能知道个大概。在你方法体里,可以初始化线程优先级,指定名称,甚至指定线程的状态(直接指定为守候线程),设置组 .act

参数:通过执行runnable变量来实例化一个线程

返回一个线程或者null.具体看你的内部实现哈!

当线程池需要创建线程的时候,就会调用这个方法;

一般自定义线程工厂的话,会指定线程名称,这样问题追溯起来也容易一些。

java 线程池中线程ID的编号 java线程池threadfactory_自定义线程池_02

2、接口定义

java 线程池中线程ID的编号 java线程池threadfactory_自定义线程池_03

1、 根据需要创建新线程的对象。使用线程工厂移除对new Thread(Runnable)(这种用法)调用的硬连线,应用程序可以使用特殊的线程子类,优先级等。

2、一个简单的接口实现样例:

class SimpleThreadFactory implements ThreadFactory {
     public Thread newThread(Runnable r) {
       return new Thread(r);
    }
   }

3、The {@link Executors#defaultThreadFactory} 示意这个链接提供了一个更有用的实现,创建的时候就设置线程的上下文(你点进去就是下图了)

java 线程池中线程ID的编号 java线程池threadfactory_System_04