多线程(2)----线程的同步互斥

本文详细探讨了多线程环境下的同步互斥问题,包括线程间通信、锁机制、信号量的应用,旨在帮助开发者理解如何避免竞态条件和死锁,确保程序的正确性和高效性。
摘要由CSDN通过智能技术生成

一、线程的安全问题                                                                                                                           点击此处返回总目录

二、同步代码块的方式,解决线程安全问题

三、同步方法的方式,解决线程安全问题

四、Lock接口的方式,解决线程安全问题

五、死锁

六、线程的等待与唤醒

 

 

 

注:关于同步互斥的概念,一定要看操作系统。

 

一、线程的安全问题

多个线程同时操作一个共享数据,可能会出现安全问题。比如下面的程序:

 

//Ticket.java

package cn.itcast.demo01;

public class Ticket implements Runnable {
    private int tickets = 3;
    
    public void run(){
        while(true){
            if(tickets>0){                                                          //1处。
                try {
                    Thread.sleep(10);                                         //首先休眠10毫秒
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"sold:"+tickets--);       //然后再卖一张票
            }
        }
    }
}

//Test.java

package cn.itcast.demo01;

public class Test {
    public static void main(String[] args) {
        Ticket ti = new Ticket();                    //创建了一个对象。成员变量tickets只有一份。下面的多个run()方法走的是同一个变量。
        
        Thread th0 = new Thread(ti);
        Thread th1 = new Thread(ti);
        Thread th2 = new Thread(ti);
        
        th0.start();th1.start();th2.start();
    }
}

运行结果:

Thread-2sold:3
Thread-0sold:2
Thread-2sold:1
Thread-1sold:0
Thread-0sold:-1
 

结果分析:

按照尝试来讲,卖出3,2,1之后就应该停止了。但是程序还多卖了两张。这就是线程不安全问题。

当线程1运行到1处时,发现还有1张票。然后就准备卖,但是还没卖。这时候线程2进来了,发现还是有1张票。于是就卖了,票数就变成了0。当线程2运行完之后,线程1继续执行,结果就导致变成了-1。

 

二、同步代码块的方式,解决线程安全问题

为了解决安全问题,java提供了同步技术,保证安全性。

 

公式:

       synchronized(同步锁){

            线程要操作的共享代码段

       }

 

同步代码块的原理:

同步锁可以是任意的对象,专业词汇叫对象监视器。

没有锁的线程不能执行,只有等待。先获取锁,然后执行,最后释放锁。

 

使用:

//Ticket.java

package cn.itcast.demo01;

public class Ticket implements Runnable {
    private int tickets = 30;
    private Object obj = new Object();
    
    public void run(){
        while(true){
            synchronized (obj) {                         //包起来的是共享代码段。这里用的是自己在类中定义一个对象,然后作为同步锁                                                                              的方式。也可以不用定义对象,直接写"this",this就是本类对象的引用。就是后面创建

                                                                    的对象"ti"。
                if(tickets>0){
                    try {
                        Thread.sleep(10);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName()+"sold:"+tickets--);
                }
            }//synchronized
        }
    }
}

//Test.java

package cn.itcast.demo01;

public class Test {
    public static void main(String[] args) {
        Ticket ti = new Ticket();
        
        Thread th0 = new Thread(ti);
        Thread th1 = new Thread(ti);
        Thread th2 = new Thread(ti);
        
        th0.start();th1.start();th2.start();
    }
}

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值