您的位置:首页 > 其它

线程的同步机制

2014-12-12 22:45 106 查看
1 线程安全问题的原因:由于一个线程在操作共享数据过程中,未执行完毕的情况下,另外的线程有参与进来,导致共享数据存在安全问题

2 解决方法:必须让一个线程操作共享数据完毕以后,其它线程才有机会参与共享数据的操作

3 java如何实现线程的安全,现成的同步机制

synchronized(同步监视器){  //需要被同步的代码块(操作共同数据的代码)}

同步监视器:由任何一个类的对象充当,哪个线程获取此监视器,就执行大括号里被同步的代码

1)同步代码块

class Window2 implements Runnable{
private int ticket = 100;  //不用static,只创建一个对象
public void run(){
while(true){
synchronized(this){ //注意一个线程进入同步代码块中,在执行的其中的代码过程中其他线程不能进入,所以称为同步锁。synchronized一定要放在共享数据前后
if(ticket>0){  //ticket是共享数据,所以同步从这个地方开始,如果放在while前,那么线程1一直占用cpu直到结束,线程2,3就进不来了


try {
Thread.currentThread().sleep(20);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}  //只能加try-catch,不能throws,因为子类方法异常不能大于父类
System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--);
}
else{
break;
}}


class Window extends Thread{
static int ticket = 100;
static Object obj = new Object();
public void run(){
while(true){
synchronized (obj) {
if (ticket > 0) {
try {
Thread.currentThread().sleep(20);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + ":"
+ ticket--);
} else {
break;
}
}


 

2)同步方法: 将操作共享数据的方法声明为synchronized,此方法称为同步方法,能够保证当其中一个线程执行此方法时,其他线程在外等待直到此线程执行完此方法。只能在实现方式中能使用,继承方式3个对象,有3个this,不能用同步方法

同步方法的锁:this,不用显式指明。

package lianxi1;
//方式二:实现方式创建多线程
class Window3 implements Runnable{
private int ticket = 100;  //不用static,只创建一个对象
public void run(){
while(true){
show();
}
}
public synchronized void show(){ //不是显式调用this,因为只有一个对象,所以只有this就是同步监视器
if(ticket>0){  //ticket是共享数据,所以从这个地方开始
try {
Thread.currentThread().sleep(20);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}  //只能加try-catch,不能throws,因为子类方法异常不能大于父类
System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--);
}
}
}
public class TestThreadWindow3 {

public static void main(String[] args) {
Window3 w2 = new Window3();
Thread t1 = new Thread(w2); //将Runnable实现类对象作为形参传人Thread构造器,执行start方法
Thread t2 = new Thread(w2);
Thread t3 = new Thread(w2);
t1.setName("窗口1");
t2.setName("窗口2");
t3.setName("窗口3");
t1.start();
t2.start();
t3.start();

}

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: