Java并发 两个线程交替执行和死锁
今天看到一个题:两个线程交替打印奇数和偶数,即一个线程打印奇数,另一个打印偶数,交替打印从1到100。想了下有多重实现方法。
wait和notify方法:
public class OddEven {
private static int i = 1;
private static Object o = new Object();
private static class MyThread extends Thread {
MyThread(String name) {
super(name);
}
public void run() {
try {
synchronized (o) {
while (i < 100) {
System.out.println(this.getName() + i++);
o.notify();
o.wait();
}
o.notify();
}
System.out.println(this.getName() + " over");
} catch (Exception e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
new MyThread("奇 ").start();
new MyThread("偶 ").start();
}
}
---
输出:
奇 1
偶 2
奇 3
偶 4
奇 5
偶 6
奇 7
...
LockSupport实现
public class OddEvenLockSupport {
private static int i = 1;
private static Thread t1 = new MyThread1();
private static Thread t2 = new MyThread2();
public static void main(String[] args) {
t1.start();
t2.start();
}
private static class MyThread1 extends Thread {
@Override
public void run() {
while (i < 51) {
System.out.println("奇 " + i++);
LockSupport.unpark(t2);
LockSupport.park();
}
System.out.println(getName() + " over");
LockSupport.unpark(t2);
}
}
private static class MyThread2 extends Thread {
@Override
public void run() {
while (i < 51) {
LockSupport.park();//顺序很重要
System.out.println("偶 " + i++);
LockSupport.unpark(t1);
}
System.out.println(getName() + " over");
LockSupport.unpark(t1);
}
}
}
---
使用Condition
public class OddEven2 {
private static Lock lock = new ReentrantLock();
static Condition even_c = lock.newCondition();
static Condition odd_c = lock.newCondition();
private static class Odd extends Thread {
public void run() {
int odd = 1;
try {
lock.lock();
while (odd < 100) {
System.out.println("奇 " + odd);
odd += 2;
even_c.signal();
odd_c.await();
}
System.out.println("奇 over");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
even_c.signal();
lock.unlock();
}
}
}
private static class Even extends Thread {
public void run() {
int even = 0;
try {
lock.lock();
while (even < 100) {
System.out.println("偶 " + even);
even += 2;
odd_c.signal();
even_c.await();
}
System.out.println("偶 over");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
odd_c.signal();
lock.unlock();
}
}
}
public static void main(String[] args) {
new Even().start();
new Odd().start();
}
}
---
另一个题,写出会导致死锁的代码:
public class DeadLock extends Thread {
private Object lock1;
private Object lock2;
public DeadLock(Object o1, Object o2) {
this.lock1 = o1;
this.lock2 = o2;
}
public void run() {
synchronized (lock1) {
try {
TimeUnit.MILLISECONDS.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (lock2) {
Q.p("end");
}
}
}
public static void main(String[] args) {
Object o1 = new Object(), o2 = new Object();
new DeadLock(o1, o2).start();
new DeadLock(o2, o1).start();
Q.p("main end");
}
}
---
基础回顾:
每个Java对象都拥有一个锁标记,即monitor(监视器),称为对象锁。
wait:
调用obj.wait()方法会阻塞当前线程,直到另一线程调用obj.notify()或obj.notifyAll()方法,其中obj为同一对象,当前线程必须持有obj对象锁,即必须在synchronized方法或代码块中,否则抛出IllegalMonitorStateException。调用wait方法的线程会释放对象锁。
notify:
唤醒一个正在等待该对象锁的线程,若有多个线程都在等待也只会唤醒一个。被唤醒的线程无法立即执行,直到当前线程放弃该对象锁,然后被唤醒的线程会像往常一样与任何其他线程竞争该对象锁,
该方法也必须在持有该对象锁的线程里调用,即必须在synchronized方法或代码块中,否则抛出IllegalMonitorStateException。
synchronized:
synchronized关键字标记一个方法或者代码块,当某个线程调用该对象的synchronized方法或者访问synchronized代码块时,这个线程便获得了该对象的锁,其他线程暂时无法访问这个方法,只有等待这个方法执行完毕或者代码块执行完毕,这个线程才会释放该对象的锁,其他线程才能执行这个方法或者代码块。
end
Java并发 两个线程交替执行和死锁的更多相关文章
- 简单的线程同步问题:两个线程交替执行N次【Synchronized、Lock、ArrayBlockingQueue】
方法一:传统的线程方法import org.apache.log4j.Logger; /** * 两个线程执行的代码片段要实现同步互斥的效果,它们必须用同一个Lock对象.<br/> * ...
- 面试必问!Java 多线程中两个线程交替执行,一个输出偶数,一个输出奇数
前言 楼主今天在面经上看到这个题,挺有意思,小小的题目对多线程的考量还挺多.大部分同学都会使用 synchronized 来实现.楼主今天带来另外两种优化实现,让你面试的时候,傲视群雄! 第一种 sy ...
- 使用Java线程并发库实现两个线程交替打印的线程题
背景:是这样的今天在地铁上浏览了以下网页,看到网上一朋友问了一个多线程的问题.晚上闲着没事就决定把它实现出来. 题目: 1.开启两个线程,一个线程打印A-Z,两一个线程打印1-52的数据. 2.实现交 ...
- Java并发编程:线程间协作的两种方式:wait、notify、notifyAll和Condition
Java并发编程:线程间协作的两种方式:wait.notify.notifyAll和Condition 在前面我们将了很多关于同步的问题,然而在现实中,需要线程之间的协作.比如说最经典的生产者-消费者 ...
- 19、Java并发编程:线程间协作的两种方式:wait、notify、notifyAll和Condition
Java并发编程:线程间协作的两种方式:wait.notify.notifyAll和Condition 在前面我们将了很多关于同步的问题,然而在现实中,需要线程之间的协作.比如说最经典的生产者-消费者 ...
- Java 并发编程:线程间的协作(wait/notify/sleep/yield/join)
Java并发编程系列: Java 并发编程:核心理论 Java并发编程:Synchronized及其实现原理 Java并发编程:Synchronized底层优化(轻量级锁.偏向锁) Java 并发编程 ...
- Java并发编程:线程池的使用
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- Java并发编程:线程池的使用(转)
Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...
- Java并发编程:线程控制
在上一篇文章中(Java并发编程:线程的基本状态)我们介绍了线程状态的 5 种基本状态以及线程的声明周期.这篇文章将深入讲解Java如何对线程进行状态控制,比如:如何将一个线程从一个状态转到另一个状态 ...
随机推荐
- INSPIRED启示录 读书笔记 - 第14章 产品评审团
制定更及时.更可靠的产品决策 制定决策通常是既耗时又费力的,产品公司需要一套机制让决策者和相关人员及时作出明智的产品决策.成立产品评审团是最好的解决途径 组织产品评审团的难点在于既要为高管制定产品决策 ...
- Go 语言为Fibonacci函数实现Read方法
Go语言非常灵活,只要为对象实现了相应的方法就可以把他看成实现了某个接口,类似于Durk Type, 为Fibonacci实现Read方法,就可以像读取文件一样,去读取下一个Fibonacci值. 示 ...
- MySQL实时性能监控工具doDBA tools
doDBA tools是什么? doDBA tools是一个基于控制台的远程监控工具,它不需要在本地/远程系统上安装任何软件,它可以实时收集操作系统.MySQL.InnoDB的实时性能状态数据,并可以 ...
- 【P2629】好消息,坏消息(前缀和+单调队列优化DP)
一激动一颓就会忘了总结... 前面的大黄题就不总结了. 这个题我只想说一声艹,一开始的思路就是正确的,然后计算的时候有了一个瑕疵,不过很快也就改过来了.然后却一直连样例都过不了.仔仔细细看了老半天,经 ...
- jsonp 实现跨域例子
直接上代码: js: <html> <head> <title>JSONP</title> </head> <script src = ...
- YII2笔记之一
安装advanced:执行init 执行yii.bat 创建数据库 修改common/config/main-local.php中的db配置 执行migratebasic:web目录是可以被外部直 ...
- OpenStack for NFV applications: enabling Single Root I/O virtualization and PCI-Passthrough
http://superuser.openstack.org/articles/openstack-for-nfv-applications-enabling-single-root-i-o-virt ...
- ie 元素兼容性总结
css 属性元素 2.z-index 正常按自身层级决定显示顺序,在ie6 7 还需要依赖于父级的层级决定,排布在后边的元素排在前面.后来者居上覆盖前者. IE6,7支持inline元素转换成inli ...
- mysql: 模糊查询 feild like keyword or feild like keyword , concat(feild1,feild2,feild3) like keyword
mysql: 模糊查询 feild like '%keyword%' or feild like'% keyword%' , 或者 concat(feild1,feild2,feild3) like ...
- TCP/IP详解学习笔记(4)-ICMP协议,ping和Traceroute【转】
转自:http://blog.csdn.net/goodboy1881/article/details/670761 1.IMCP协议介绍 前面讲到了,IP协议并不是一个可靠的协议(是一种尽力传送的协 ...