多线程经典题——顺序打印1-75
今天学习了线程安全,碰到了这个题目:
启动3个线程打印递增的数字, 线程1先打印1,2,3,4,5, 然后是线程2打印6,7,8,9,10,
然后是线程3打印11,12,13,14,15. 接着再由线程1打印16,17,18,19,20....
以此类推, 直到打印到75. 程序的输出结果应该为:
线程1: 1
线程1: 2
线程1: 3
线程1: 4
线程1: 5
线程2: 6
线程2: 7
线程2: 8
线程2: 9
线程2: 10
...
线程3: 71
线程3: 72
线程3: 73
线程3: 74
线程3: 75
这道题可以用三个自定义线程类做,但是觉得太麻烦了,而且没有用到synchronized修饰,所以如果通过对象锁的思路是:主线程中放一个循环1-3调用自己定义的一个线程类的start方法,在自定义的类中 构造函数接参,一个是共同的对象object(就用这个对象实现锁)还有一个是i循环的值,因为这题需要按顺序循环打印,就需要判断是否是顺序的,而需要判断就需要找规律,线程三次一循环,一次输出5个,那么在自定义中可以定义一个count来计数,一个打印五次后,count++,然后对3取余,如果值与传入的循环i值相等,则运行打印,不然则调用wait方法,等待应该按顺序执行的i值。
public class work3 {
public static void main(String[] args) {
Object obj = new Object();
for(int i=1;
i<=3;
i++){//传入对象锁 和 i值new Thread(new MyThread2(obj,i), "线程"+i).start();
}
}
}
class MyThread2 implements Runnable{
private static int i = 0;
//打印的值 1-75
private static int count=0;
//计数 三次一轮回
private Object obj;
private int n;
//接参 i值
public MyThread2(Object obj,int n) {
this.obj=obj;
this.n = n;
}
@Override
public void run() {
synchronized (obj) {
while(i<75){//i++ 在代码块 所以到74就可以了obj.notifyAll();
//唤醒所有线程if(count%3!=(n-1)){ //找出下一个线程不正确的线程等待try {
obj.wait();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
i++;
System.out.println(Thread.currentThread().getName()+": "+i);
if(i%5 == 0){ //打印了五次后 此线程让出资源,等待
try {
count++;
//count是static修饰 ,为了共享
System.out.println();
obj.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
}
由这题也可以想到另一到多线程题,开启3个线程,这3个线程的ID分别为A、B、C,
* 每个线程将自己的ID在屏幕上打印10遍,要求输出结果必须按ABC的顺序显示;
* 如:ABCABC….依次递推。 顺序输出ABC用synchronized的代码。
上面的代码改动下,不用每次打印五次,只要打印一次,在循环十次
【多线程经典题——顺序打印1-75】
package work;
public class work1_2 {
public static void main(String[] args) {
Object obj = new Object();
for(int i =0;
i<3;
i++){
new Thread(new myThreads(obj,""+(char)(i+65),i)).start();
}
}
}
class myThreads implements Runnable{
private Object obj;
private String name;
private int i;
private static int count = 0;
public myThreads(Object obj,String name, int i) {
this.obj = obj;
this.name = name;
this.i = i;
} @Override
public void run() {
synchronized (obj) {
for(int n=0;
n<10;
n++){
obj.notifyAll();
if((count)%3!=i){
try {
obj.wait();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
System.out.println(name);
try {
count++;
obj.wait();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
}
当然,这个打印ABC的有更简单的方法,就是sleep,写三个自定义的线程类,A类sleep时间最少,C最多,这样非常容易就实习了。
推荐阅读
- 放屁有这三个特征的,请注意啦!这说明你的身体毒素太多
- 爱就是希望你好好活着
- 昨夜小楼听风
- 知识
- 死结。
- 我从来不做坏事
- 烦恼和幸福
- 关于QueryWrapper|关于QueryWrapper,实现MybatisPlus多表关联查询方式
- Linux下面如何查看tomcat已经使用多少线程
- 说得清,说不清