多线程练习题

多线程交替打印1~10的奇偶数


思路

  1. 搞两条线程,一条线程打印奇数任务,一条线程打印偶数任务。
  2. 为了防止线程间的无序争抢,使用synchronized锁。
  3. 为了保证交互打印,使用wait/notify,打印完一条先等待,让别的线程打印。以此循环。

实现1:synchronized方法对象锁

public class MyTask {

    public synchronized void printNumber(int i) {
        try {
            this.notify();
            System.out.println(Thread.currentThread().getName() + " " + i);
            this.wait();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
public class SwapPrint {

    public static void main(String[] args) {
        final MyTask myTask = new MyTask();
        
        Thread t1 = new Thread(new Runnable() {

            @Override
            public void run() {
                for (int i = 1 ; i <= 10 ; i+=2) {
                    myTask.printNumber(i);
                }
            }
        });
        
        t1.setName("Thread t1");
        t1.start();
        
        Thread t2 = new Thread(new Runnable() {

            @Override
            public void run() {
                for (int i = 2 ; i <= 10 ; i+=2) {
                    myTask.printNumber(i);
                }
            }
        });
        
        t2.setName("Thread t2");
        t2.start();
    }
}

思考:
有的人会搞不懂业务方法里面上来先notify,然后结束调用wait?

  1. 首先调用notify方法,将另一个线程唤醒。但是另一个线程根本进不来当前的方法(因为有synchronized锁)。
  2. 后面调用wait方法,将当前线程等待,wait方法会释放锁,另一个线程就可以进来了。

实现2:synchronized(this)对象锁

这种实现和实现1没什么区别,实现1虽然是synchronized修饰方法的方式,实际上还是使用当前对象的锁。

public class MyTask {

    public void printNumber(int i) {
        try {
            synchronized (this) {
                this.notify();
                System.out.println(Thread.currentThread().getName() + " " + i);
                this.wait();
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

思考:
synchronized关键字只是控制当前代码区域,只能有一个线程进入。
wait/notify方法可以让当前线程等待,也可以唤醒其他线程。

实现3:等待队列Condition实现

public class MyTask {

    private ReentrantLock rl = new ReentrantLock();
    private Condition condition = rl.newCondition();
    
    public void printNumber(int i) {
        try {
            rl.lock();
            condition.signal();
            System.out.println(Thread.currentThread().getName() + " " + i);
            condition.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            rl.unlock();
        }
    }
}

思考:

  1. 调用lock方法,当前线程设置同步状态,加入到aqs同步队列中,获取重入锁。
  2. 调用signal方法,唤醒等待队列中的线程,然后加入到aqs同步队列。
  3. 调用await方法,将当前线程加入到aqs等待队列。

ABC三个线程如何保证顺序执行


题目

ABC三个线程如何保证顺序执行。三个线程同时启动,然后按照顺序执行,每个线程执行10次。

思路

首先想到了等待队列Condition唤醒部分线程,使用ReentrantLock进行加锁。

初始版实现

/**
 * @description A\B\C三个线程顺序执行10次
 * 
 * @author sunpy
 * @date 2018年11月28日  下午2:23:45
 */
public class MyTest {

    static class MyTask {
        private static ReentrantLock rl = new ReentrantLock();
        private static Condition conditionA = rl.newCondition();
        private static Condition conditionB = rl.newCondition();
        private static Condition conditionC = rl.newCondition();
        private static int number = 0;
    
        public void execute() {
            rl.lock();
            
            try {
                while (number < 30) {
                    if (number % 3 == 0) {
                        System.out.println(Thread.currentThread().getName() + " - " + number);
                        number++;
                        conditionB.signal();
                        conditionA.await();
                    }
                    
                    if (number % 3 == 1) {
                        System.out.println(Thread.currentThread().getName() + " - " + number);
                        number++;
                        conditionC.signal();
                        conditionB.await();
                    }
                    
                    if (number % 3 == 2) {
                        System.out.println(Thread.currentThread().getName() + " - " + number);
                        number++;
                        conditionA.signal();
                        conditionC.await();
                    }
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                rl.unlock();
            }
        }
    }
    
    
    public static void main(String[] args) {
        final MyTask myTask = new MyTask();
        new Thread(new Runnable() {

            @Override
            public void run() {
                myTask.execute();
            }
        }, "A").start();
        Thread.sleep(1000);
        new Thread(new Runnable() {

            @Override
            public void run() {
                myTask.execute();
            }
        }, "B").start();
        Thread.sleep(1000);
        new Thread(new Runnable() {

            @Override
            public void run() {
                myTask.execute();
            }
        }, "C").start();
        
    }
}

说明:通过余数判断来执行,需要循环30次,有点不合理。想了下,执行10次,循环10次,但是让线程按照ABC执行就可以了。根据这个想法写出了改进版。

改进版实现

/**
 * @description A\B\C三个线程顺序执行10次
 * 
 * @author sunpy
 * @date 2018年11月28日  下午2:23:45
 */
public class MyTest {

    static class MyTask {
        private static ReentrantLock rl = new ReentrantLock();
        private static Condition conditionA = rl.newCondition();
        private static Condition conditionB = rl.newCondition();
        private static Condition conditionC = rl.newCondition();
    
        public void execute(String flag) {
            rl.lock();
            
            try {
                for (int i = 1 ; i <= 10 ; i++) {
                    if ("A".equals(flag)) {
                        System.out.println(Thread.currentThread().getName() + " - " + i);
                        conditionB.signal();
                        conditionA.await();
                    }
                    
                    if ("B".equals(flag)) {
                        System.out.println(Thread.currentThread().getName() + " - " + i);
                        conditionC.signal();
                        conditionB.await();
                    }
                    
                    if ("C".equals(flag)) {
                        System.out.println(Thread.currentThread().getName() + " - " + i);
                        conditionA.signal();
                        conditionC.await();
                    }
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                rl.unlock();
            }
        }
    }
}

Thread.currentThread.getName和this.getName的区别


this.getName

this的意思是代表当前对象的。而this在线程的环境下,代表的是当前线程实例对象本身。所以this.getName是当前线程实例对象的线程名称是什么。

Thread.currentThread.getName

Thread.currentThread.getName意思是在当前代码块中执行的线程名称是什么。

例子

public class CountOperate extends Thread {
    
    public CountOperate() {
        System.out.println("CountOperate---begin");
        System.out.println("Thread.currentThread().getName()=" + Thread.currentThread().getName());
        System.out.println("Thread.currentThread().isAlive()=" + Thread.currentThread().isAlive());
        System.out.println("this.getName()=" + this.getName());
        System.out.println("this.isAlive()=" + this.isAlive());
        System.out.println("CountOperate---end");
    }

    @Override
    public void run() {
        System.out.println("run---begin");
        System.out.println("Thread.currentThread().getName()=" + Thread.currentThread().getName());
        System.out.println("Thread.currentThread().isAlive()=" + Thread.currentThread().isAlive());
        System.out.println("this.getName()=" + this.getName());
        System.out.println("this.isAlive()=" + this.isAlive());
        System.out.println("run---end");
    }
}

public class Test {
    
    public static void main(String[] args) {
        CountOperate c = new CountOperate();
        Thread t1 = new Thread(c);
        System.out.println("main begin t1 isAlive=" + t1.isAlive());
        t1.start();
        System.out.println("main end t1 isAlive=" + t1.isAlive());
    }
}

结果:

CountOperate---begin
Thread.currentThread().getName()=main
Thread.currentThread().isAlive()=true
this.getName()=Thread-0
this.isAlive()=false
CountOperate---end
main begin t1 isAlive=false
main end t1 isAlive=true
run---begin
Thread.currentThread().getName()=Thread-1
Thread.currentThread().isAlive()=true
this.getName()=Thread-0
this.isAlive()=false
run---end

说明:
① 首先搞明白该例子中一共有三条线程:
main线程:执行main方法的线程。
Thread-0线程:当前CountOperate线程实例的线程。
Thread-1线程:new Thread类创建的线程。


② CountOperate初始化代码说明:

CountOperate---begin
Thread.currentThread().getName()=main
Thread.currentThread().isAlive()=true
this.getName()=Thread-0
this.isAlive()=false
CountOperate---end

执行构造器,然后当前main方法中执行初始化CountOperate类,所以当前线程的名称是main,状态是true。而当前CountOperate线程实例为Thread-0,没有启动所以状态为false。


③ Thread1线程说明:

main begin t1 isAlive=false
main end t1 isAlive=true

Thread-1线程的线程执行器没有启动,那么状态为false。执行start方法,Thread-1线程的线程执行器启动,那么状态就为true了。


④ run方法执行说明:

run---begin
Thread.currentThread().getName()=Thread-1
Thread.currentThread().isAlive()=true
this.getName()=Thread-0
this.isAlive()=false
run---end

首先执行run方法的线程为Thread-1,所以当前代码块中的线程为Thread1(Thread.currentThread().getName()=Thread-1),状态自然为true。
但是this.getName()是代表当前线程Thread-0的实例,因为CountOperate线程根本没有启动,所以状态为false。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,924评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,781评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,813评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,264评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,273评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,383评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,800评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,482评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,673评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,497评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,545评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,240评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,802评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,866评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,101评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,673评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,245评论 2 341

推荐阅读更多精彩内容

  • 该文章转自:http://blog.csdn.net/evankaka/article/details/44153...
    加来依蓝阅读 7,328评论 3 87
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,948评论 1 18
  • 林炳文Evankaka原创作品。转载自http://blog.csdn.net/evankaka 本文主要讲了ja...
    ccq_inori阅读 643评论 0 4
  • 写在前面的话: 这篇博客是我从这里“转载”的,为什么转载两个字加“”呢?因为这绝不是简单的复制粘贴,我花了五六个小...
    SmartSean阅读 4,708评论 12 45
  • 爱的艺术 当爱已成往事 记忆已化成田野上 踽踽独行的脚印 过去已变成永恒 未来图将好景 在茫茫的宇宙间 地球照样运...
    郭大牛阅读 330评论 2 15