【java并发】线程并发库的使用

1. 线程池的概念

  在java5之后,就有了线程池的功能了,在介绍线程池之前,先来简单看一下线程池的概念。假设我开了家咨询公司,那么每天会有很多人过来咨询问题,如果我一个个接待的话,必然有很多人要排队,这样效率就很差,我想解决这个问题,现在我雇几个客服,来了一个咨询的,我就分配一个客服去接待他,再来一个,我再分配个客服去接待……如果第一个客服接待完了,我就让她接待下一个咨询者,这样我雇的这些客服可以循环利用。这些客服就好比不同的线程,那么装这些线程的容器就称为线程池。

2. Executors的使用

  Executors工具类是用来创建线程池的,这个线程池可以指定线程个数,也可以不指定,也可以指定定时器的线程池,它有如下常用的方法:

方法名 作用
newFixedThreadPool(int nThreads) 创建固定数量的线程池
newCachedThreadPool() 创建缓存的线程池
newSingleThreadExecutor() 创建单个线程
newScheduledThreadPool(int corePoolSize) 创建定时器线程池

2.1 固定数量的线程池

  先来看下Executors工具类的使用:

public class ThreadPool {
//线程池的概念与Executors类的使用
    public static void main(String[] args) {
        //固定线程池:创建固定线程数去执行线程的任务,这里创建三个线程
        ExecutorService threadPool = Executors.newFixedThreadPool(3);

        for (int i = 1; i <= 10; i++) {//向池子里扔10个任务
            final int task = i;
            threadPool.execute(new Runnable() {//execute方法表示向池子中扔任务,任务即一个Runnable

                @Override
                public void run() {
                    for (int j = 1; j <= 5; j++) {
                        try {
                            Thread.sleep(20);
                        } catch (InterruptedException e) {
                            // TODO Auto-generated catch block
                            e.printStackTrace();
                        }
                        System.out.println(Thread.currentThread().getName()
                                + " looping of " + j + " for task of " + task);
                    }
                }
            });
        }
        System.out.println("all of 10 tasks have committed!");
        threadPool.shutdown(); //执行完任务后关闭
//      threadPool.shutdownNow(); //立即关闭
    }
}

  从代码中可以看出,有了Executors工具类,我们创建固定数量的线程数就方便了,这些线程都去做同样的任务。threadPool.execute表示从池子里取出一个线程去执行任务,上面定义了三个线程,所以每次会取三个任务去让线程执行,其他任务等待,执行完后,再从池子里取三个任务执行,执行完,再取三个任务,最后一个任务三个线程有一个会抢到执行。所以定义了线程数量的话,每次会执行该数量的任务,因为一个线程一个任务,执行完再执行其他任务。

  因为这个执行结果有点多,就不贴结果了,反正每次三个任务一执行,直到执行完10个任务为止。

2.2 缓存线程池

  所谓缓存线程池,指的是线程数量不固定,一个任务来了,我开启一个线程为其服务,两个任务我就开启两个,N个任务我就开启N个线程为其服务。如果现在只剩1个任务了,那么一段时间后,就把多余的线程给干掉,保留一个线程为其服务。所以可以改写一下上面的代码:

public class ThreadPool {
//线程池的概念与Executors类的使用
    public static void main(String[] args) {
        //缓存的线程池
        //自动根据任务数量来设定线程数去服务,多了就增加线程数,少了就减少线程数
        //这貌似跟一般情况相同,因为一般也是一个线程执行一个任务,但是这里的好处是:如果有个线程死了,它又会产生一个新的来执行任务
        ExecutorService threadPool = Executors.newCachedThreadPool();
        for (int i = 1; i <= 10; i++) {//扔5个任务
            final int task = i;
            threadPool.execute(new Runnable() {//向池子中扔任务,任务即一个Runnable

                @Override
                public void run() {
                    for (int j = 1; j <= 5; j++) {
                        try {
                            Thread.sleep(20);
                        } catch (InterruptedException e) {
                            // TODO Auto-generated catch block
                            e.printStackTrace();
                        }
                        System.out.println(Thread.currentThread().getName()
                                + " looping of " + j + " for task of " + task);
                    }
                }
            });
        }
        System.out.println("all of 10 tasks have committed!");
        threadPool.shutdown(); //执行完任务后关闭
    }
}

  使用缓存线程池的时候,会自动根据任务数量来产生线程数,即线程跟着任务走。运行结果也不贴了,有点多。

  那么创建单个线程池newSingleThreadExecutor()就写了,把上面那个方法改掉就行了,就只有一个线程去执行10个任务了,但是这跟我们平常直接new一个线程还有啥区别呢?它还有个好处就是,如果线程死了,它会自动再生一个,而我们自己new的就不会了。如果线程死了还要重新产生一个,也就是说要保证有一个线程在执行任务的话,那么newSingleThreadExecutor()是个很好的选择。

3. 线程池启动定时器

  我们可以用静态方法newScheduledThreadPool(int corePoolSize)来定义一个定时器线程池,可以指定线程个数。然后再调用schedule方法,传进去一个Runnable和定时时长即可,见代码:

public class ThreadPool {

    public static void main(String[] args) {
        Executors.newScheduledThreadPool(3).schedule(new Runnable() {

            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName() + " bombing");
            }
        }, 2, TimeUnit.SECONDS);
    }
}

  定义了三个线程,会有一个率先抢到任务执行在2秒后执行。这只是创建了一个任务,如果我们要使用这个线程池去执行多个任务咋办呢?schedule中只能传入一个Runnable,也就是说只能传入一个任务,解决办法跟上面那些程序一样,先拿到创建的线程池,再循环多次执行schedule,每次都传进去一个任务即可:

public class ThreadPool {

    public static void main(String[] args) {
        //拿到定时器线程池
        ScheduledExecutorService threadPool = Executors.newScheduledThreadPool(3);
        for(int i = 1; i <= 5; i ++) { //执行5次任务
            threadPool.schedule(new Runnable() {

                @Override
                public void run() {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName() + " bombing");
                }
            }, 2, TimeUnit.SECONDS);
        }
    }
}

  因为线程池中只有3个线程,但是有5个任务,所以会先执行3个任务,剩下两个任务,随机2个线程执行,看下结果:

pool-1-thread-3 bombing

pool-1-thread-2 bombing

pool-1-thread-1 bombing

pool-1-thread-2 bombing

pool-1-thread-3 bombing

  如果我想5秒后执行一个任务,然后每个2秒执行一次该怎么办呢?我们可以调用另一个方法scheduleAtFixedRate,这个方法中传进去一个Runnable,一个定时时间和每次重复执行的时间间隔,如下:

public class ThreadPool {

    public static void main(String[] args) {
        Executors.newScheduledThreadPool(3).scheduleAtFixedRate(new Runnable() {

            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName() + " bombing");
            }
        }, 5, 2, TimeUnit.SECONDS);
    }
}

  这样就可以5秒后执行,并且以后每隔2秒执行一次了。这个方法有个瑕疵,就是无法设置指定时间点执行,官方JDK提供的解决办法是data.getTime()-System.currentTimeMillis()来获取相对时间,然后放到上面方法的第二个参数即可。

  线程并发库的使用就总结这么多吧~

  相关阅读:http://blog.csdn.net/column/details/bingfa.html



—–乐于分享,共同进步!

—–更多文章请看:http://blog.csdn.net/eson_15

时间: 2024-08-26 22:42:32

【java并发】线程并发库的使用的相关文章

【Java】线程并发拷贝程序

据说大连某211高校的信息学院的李教授非常好这口,他带的每个操作系统本科班,每个学期都必须完成这个程序,不过网上关于这方面的资料甚少,就只有一份C语言版. 然而,那份被历届学生已经抄烂,改实验结果把李教授忽悠了N年的C语言版,所使用的类.所开的线程与进程也不甚合理,把一个本来非常简单的程序搞得十分冗长.明明这个线程并发拷贝程序只涉及到线程的互斥的方面,与线程的同步半点关系,这个线程并发拷贝程序的互斥,也就是每个线程操作的文件只能是一个,保证不出现一个文件被多个文件操作的情况即可,做完拷贝之后根本

【Java】线程并发、互斥与同步

网络上对于线程的解析总是天花龙凤的,给你灌输一大堆概念,考研.本科的操作系统必修课尤甚,让你就算仔细看完一大堆文章都不知道干什么. 下面不取网站复制粘贴,在讲解自己的Java线程并发.互斥与同步之前先给大家解构操作系统书中那些给出书者为了出书者而写的废话到底是什么意思. 大神们如果只想看程序,可以自行跳过,反正我的文章从来新手向,不喜勿看. 一.线程的基本概念 其实线程的概念非常简单,多一个线程,就多了一个做事情的人. 比如搬东西,搬10箱货物,这就是所谓的一个进程,一个人,就是所谓的一个线程,

11.9-全栈Java笔记: 线程并发协作(生产者/消费者模式)

多线程环境下,我们经常需要多个线程的并发和协作.这个时候,就需要了解一个重要的多线程并发协作模型"生产者消费者模式". 什么是生产者? 生产者指的是负责生产数据的模块(这里模块可能是:方法.对象.线程.进程). 什么是消费者? 消费者指的是负责处理数据的模块(这里模块可能是:方法.对象.线程.进程). 什么是缓冲区? 消费者不能直接使用生产者的数据,它们之间有个"缓冲区".生产者将生产好的数据放入"缓冲区",消费者从"缓冲区"

java基础--线程并发

线程的使用方法: public static void main(String[] args) {     Runnable r = ()->{         while (true) {             System.out.println("this is thread");         }     };     Thread t = new Thread(r);     t.start();     while (true) {         System.

【Linux】线程并发拷贝程序

据说大连某211高校的李教授越来越重口.不仅延续要求他所带的每个本科班.都要写一份线程并发拷贝程序的传统,并且还開始规定不能用Java语言写作.导致我之前写的<[Java]线程并发拷贝程序>(点击打开链接)作废.全部李教授旗下的学生,必须在毫无图形界面的Linux系统.用里面vi去写作. 这更让莘莘学子们感觉本来头来就不光明的天空更加黑暗起来. 更重要的是.若干年过去了,网上对其的研究与资料,依然是少数.依然是那份流传已久,以讹传讹的C语言版. 尽管这个程序毫无研究价值,可是本着治病救人.同一

【Java】线程管道通讯

很多操作系统对于管道的吹嘘往往是天花龙凤, 好心点的就贴段伪代码给你看,为出书而出书的,就直接一堆概念堆在上面,让人根本看不懂, 如此简单的概念,明明几句话就解释清楚,有的书还专门开出一章来讨论这个问题,完全没有必要! 一.基本概念 其实管道的概念非常简单,就是连接两个线程通讯的缓冲区,画个图就更加明白了 写者进程把自己的数据通过管道输出流写入管道,读者进程再从管道通过管道输入流拿管道里面的数据 当然进程与进程之间传递数据未必通过这个方式去传递数据, 完全可以在一个进程中设置一个public变量

Linux-Shell-使用mkfifo实现多任务并发及并发数控制

默认的情况下,Shell脚本中的命令是串行执行的,必须等到前一条命令执行完后才执行接下来的命令,但是如果我有一大批的的命令需要执行,而且互相又没有影响的情况下(有影响的话就比较复杂了),那么就要使用命令的并发执行了. 如下: #!/bin/bash IPLIST=/home/meta/ipinfo/iplist for i in $(cat ${IPLIST} |grep -viE "^#|备机|ts"|awk '{print $1}') do ssh $i "cd ~/up

Java多线程与并发库高级应用之线程数据交换Exchanger

JDK1.5提供了Exchanger用于两个线程的数据交换.两个线程先后到达交换点,先到达的线程会等待后到达的线程,然后两个线程互相交换数据,交换后双方持对方的数据. Exchanger只提供了一个构造器: Exchanger():创建一个新的Exchanger. Exchanger中也只有两个方法: V exchange(V x): 等待另一个线程到达此交换点(除非当前线程被中断),然后将给定的对象传送给该线程,并接收该线程的对象. V exchange(V x, long timeout,

Java多线程与并发库高级应用-java5线程并发库

java5 中的线程并发库 主要在java.util.concurrent包中 还有 java.util.concurrent.atomic子包和java.util.concurrent.lock子包