您的位置:首页 > 编程语言 > Java开发

java线程安全与死锁

2016-11-24 20:10 155 查看
线程 安全问题的解决方案:sun提供了线程同步机制让我们解决这类问题的。

出现线程安全问题的根本原因:
1. 存在两个或者两个以上 的线程对象,而且线程之间共享着一个资源。
2. 有多个语句操作了共享资源。
java线程同步机制的方式:



方式一:同步代码块

同步代码块的格式:

synchronized(锁对象){
需要被同步的代码...
}

同步代码块要注意事项:
1. 任意的一个对象都可以做为锁对象。
2. 在同步代码块中调用了sleep方法并不是释放锁对象的。
3. 只有真正存在线程安全问题的时候才使用同步代码块,否则会降低效率的。
4. 多线程操作的锁 对象必须 是唯一共享 的。否则无效。

方式二:同步函数  :  同步函数就是使用synchronized修饰一个函数。

同步函数要注意的事项 :
1. 如果是一个非静态的同步函数的锁 对象是this对象,如果是静态的同步函数的锁 对象是当前函数所属的类的字节码文件(class对象)。
2. 同步函数的锁对象是固定的,不能由你来指定 的。

推荐使用: 同步代码块。
原因:
1. 同步代码块的锁对象可以由我们随意指定,方便控制。同步函数的锁对象是固定 的,不能由我们来指定。

2. 同步代码块可以很方便控制需要被同步代码的范围,同步函数必须是整个函数 的所有代码都被同步了。

class SaleTicket extends Thread{

static int num = 50;//票数  非静态的成员变量,非静态的成员变量数据是在每个对象中都会维护一份数据的。

static	Object o = new Object();

public SaleTicket(String name) {
super(name);
}

@Override
public void run() {
while(true){
//同步代码块
synchronized (o) {
if(num>0){
System.out.println(Thread.currentThread().getName()+"售出了第"+num+"号票");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
num--;
}else{
System.out.println("售罄了..");
break;
}
}

}
}

}

public class Demo4 {

public static void main(String[] args) {
//创建三个线程对象,模拟三个窗口
SaleTicket thread1 = new SaleTicket("窗口1");
SaleTicket thread2 = new SaleTicket("窗口2");
SaleTicket thread3 = new SaleTicket("窗口3");
//开启线程售票
thread1.start();
thread2.start();
thread3.start();

}

}


需求: 一个银行账户5000块,两夫妻一个拿着 存折,一个拿着卡,开始取钱比赛,每次只能取一千块,要求不准出现线程安全问题。

class BankThread extends Thread{

static	int count = 5000;

public BankThread(String name){
super(name);
}

@Override  //
public synchronized  void run() {
while(true){
synchronized ("锁") {
if(count>0){
System.out.println(Thread.currentThread().getName()+"取走了1000块,还剩余"+(count-1000)+"元");
count= count - 1000;
}else{
System.out.println("取光了...");
break;
}
}
}
}

//静态的函数---->函数所属 的类的字节码文件对象--->BankThread.class  唯一的。
public static synchronized  void getMoney(){

}

}

public class Demo1 {

public static void main(String[] args) {
//创建两个线程对象
BankThread thread1 = new BankThread("老公");
BankThread thread2 = new BankThread("老婆");
//调用start方法开启线程取钱
thread1.start();
thread2.start();

}

}
运行结果:

老公取走了1000块,还剩余4000元

老婆取走了1000块,还剩余3000元

老婆取走了1000块,还剩余2000元

老婆取走了1000块,还剩余1000元

老婆取走了1000块,还剩余0元

取光了...

取光了...

java中同步机制解决了线程安全问题,但是也同时引发死锁现象。

死锁现象出现 的根本原因:
1. 存在两个或者两个以上的线程。
2. 存在两个或者两个以上的共享资源。

死锁现象的解决方案: 没有方案。只能尽量避免发生而已。

class DeadLock extends Thread{

public DeadLock(String name){
super(name);
}

public void run() {
if("张三".equals(Thread.currentThread().getName())){
synchronized ("遥控器") {
System.out.println("张三拿到了遥控器,准备 去拿电池!!");
synchronized ("电池") {
System.out.println("张三拿到了遥控器与电池了,开着空调爽歪歪的吹着...");
}
}
}else if("狗娃".equals(Thread.currentThread().getName())){
synchronized ("电池") {
System.out.println("狗娃拿到了电池,准备去拿遥控器!!");
synchronized ("遥控器") {
System.out.println("狗娃拿到了遥控器与电池了,开着空调爽歪歪的吹着...");
}
}

}
}

}


运行结果:

张三拿到了遥控器,准备 去拿电池!!

狗娃拿到了电池,准备去拿遥控器!!
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息