多线程设置优先级方法

本文介绍了如何确保多个线程执行顺序的三种方法:利用线程的优先级特性,虽然不能保证绝对顺序,但能增加概率;使用ExecutorService的newSingleThreadExecutor创建的线程池,保证先提交的线程先执行;以及通过CountDownLatch和CyclicBarrier的同步机制来控制线程执行顺序。示例代码展示了这些方法的实际效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

之前看到个问题,如果有3个线程,如何保证3个线程的执行先后顺序。其实实现的方法有很多,这里主要介绍几种。
首先,线程Thread有一个设置优先级的属性Priority,默认数值为5,设置的越高优先级越高,最高为10。但是这里需要注意的是:
优先级作用只是表示了执行的概率,也就是说优先级6的线程比5的线程概率更高,但不代表一定就优先执行。
(一)首先说下第一种方法,线程池ExecutorService有一个工厂方法newSingleThreadExecutor(),该方法生成的是单例线程,因此如果向其提交线程时,先提交的先执行,后提交的执行,以此就能实现执行顺序。
代码:

ExecutorService es = Executors.newSingleThreadExecutor();
        es.submit(new Runnable() {
            @Override
            public void run() {
                System.out.println("C");
            }
        });
        es.submit(new Runnable() {
            @Override
            public void run() {
                System.out.println("B");
            }
        });
        es.submit(new Runnable() {
            @Override
            public void run() {
                System.out.println("A");
            }
        });
        es.shutdown();

结果:C B A
(二)concurrent包里面有个CountDownLatch计数器,这里面有2个方法:c.countDown():作用是计数器减1,c1.await(),如果计数器暂时没有到0,则阻塞线程。
代码:

 System.out.println("begin");
        CountDownLatch c1 = new CountDownLatch(1);
        CountDownLatch c2 = new CountDownLatch(1);
        CountDownLatch call = new CountDownLatch(3);
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    c2.await();
                    System.out.println("A");
                    call.countDown();
                }catch (Exception e){
                    e.printStackTrace();
                }
            }
        });
        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    c1.await();
                    System.out.println("B");
                    c2.countDown();
                    call.countDown();
                }catch (Exception e){
                    e.printStackTrace();
                }
            }
        });
        Thread t3 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("C");
                c1.countDown();
                call.countDown();
            }
        });
        t1.start();
        t2.start();
        t3.start();
        try {
            call.await();
        }catch (Exception e){
            e.printStackTrace();
        }

结果 :C B A
(三)栅拦CyclicBarrier
CyclicBarrier同样也是concurrent包里的类,有2个构造函数,并且有一个await()方法,其目的是所有调用await()方法的线程都会被CyclicBarrier拦截,直到拦截的线程数量达到了CyclicBarrier的要求,然后统一一起执行。
代码:

     public static void main(String[] args) {
        CyclicBarrier cb = new CyclicBarrier(4);
        ExecutorService es = Executors.newCachedThreadPool();
        for (int i=0;i<4;i++) {
            es.submit(new RClass(cb,i));
        }
        es.shutdown();
    }
}

class RClass implements Runnable{
    private CyclicBarrier cb;
    private Integer count;
    RClass(CyclicBarrier cb,Integer count){
        this.cb = cb;
        this.count = count;
    }
    @Override
    public void run() {
        System.out.println("begin:"+count);
        try {
            cb.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (BrokenBarrierException e) {
            e.printStackTrace();
        }
        System.out.println("after:"+count);
    }

显示的结果:
begin:0
begin:1
begin:2
begin:3
after:3
after:0
after:1
after:2

这里需要注意的是:1、如果线程数量小于栅栏设置的数量,则所有线程都会一直阻塞;2、如果线程的数量大于设置的数量n,则会先把这N个线程执行了,剩下的继续拦截,除非剩下的又能满足N。从这也可以栅栏的好处,设置的CyclicBarrier 可以重复使用。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值