重点提示:
线程的本质上只是一个壳子,真正的逻辑其实在“竞态条件”中。
举个例子,比如本题中的打印,那么在竞态条件中,我只需要一个方法即可;
假如我的需求是2个线程,一个+1,一个-1,那么就是2个方法。
1、定义竞态条件中的类NumCounter
package com.maizijf.test.test.doublethread;
/**
* Created by Germmy on 2018/6/21.
*/
public class NumCounter implements Runnable {
int i;
@Override
public void run() {
for(;i<100;){
printNum();
}
System.exit(0);//当i为101时,当另一个线程主动结束这个罪恶吧,哈哈
}
private synchronized void printNum(){
i++;
System.out.println(Thread.currentThread().getName()+" is running "+i);
this.notify();
try {
this.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
注意:
1)、这里用到一个技巧,就是在竞态条件上实现Runnable接口,这样就可以少写一个线程类
2)、在线程2执行到101时,此时线程2不会进入for循环,因此它将销毁,但是它无法再notify线程1了,而且线程1此时处于睡眠状态,将长眠于此,程序不会退出,因此要想退出,必须强制退出
2、创建测试类
package com.maizijf.test.test.doublethread;
import org.junit.Test;
/**
* Created by Germmy on 2018/6/21.
*/
public class DoubleThreadTest {
@Test
public void testDoubleThreadTest(){
NumCounter numCounter=new NumCounter();
Thread t1=new Thread(numCounter);
Thread t2=new Thread(numCounter);
t1.start();
t2.start();
try {
t1.join();
t2.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("main ends...");
}
}
注意:执行完后,发现有时运行到48,有时运行到23就停止了,是因为主线程跑完就结束了,不会等t1,t2运行完,要想等t1,t2运行完,需要将t1,t2的优先级提升,且要运行完成后,main才能继续运行,这样保证了t1,t2肯定能运行完
执行结果如下