03线程同步机制

线程同步方式

一、概述

在这里插入图片描述

1.多线程访问了共享数据就会产生不安全的问题。下面是实例:
public class DemoTicket {
    public static void main(String[] args) {
        RunnableImpl impl =new RunnableImpl();
        Thread t0=new Thread(impl);
        Thread t1=new Thread(impl);
        Thread t2=new Thread(impl);
        t0.start();
        t1.start();
        t2.start();
    }
}
public class RunnableImpl implements Runnable {
    private int ticket =100;
    @Override
    public void run() {
        while(true){
            if(ticket>0){
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"正在卖>>"+ticket);
                ticket--;
            }
        }

    }
}
2.线程安全问题产生原因

在这里插入图片描述

二、解决线程安全问题:同步技术

1.同步代码块
synchronized(同步锁){
    可能会出现线程安全问题的代码(访问了共享数据的代码);
}

注意:

  • 同步到代码块中的锁对象,可以使用任意对象
  • 但是必须保证多个线程使用的锁对象是同一个
  • 锁对象作用:把同步代码块锁住,只让一个线程在同步代码块中执行
public class RunnableImpl implements Runnable {
    private int ticket =100;
    private Object obj = new Object();
    @Override
    public void run() {
        while(true){
            synchronized (obj){//同步代码块
                if(ticket>0){
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName()+"正在卖>>"+ticket);
                    ticket--;
                }
            }
            }
    }
2.同步方法

使用步骤:

  1. 把使用了共享数据导入额代码抽取出来,放到一个方法中。
  2. 在方法中添加synchronized修饰符
修饰符 synchronize void  方法名(){}
public class RunnableImpl implements Runnable {
    private int ticket =100;
   
    @Override
    public void run() {
        while(true){
             payTicked();
            }
    }
    public synchronized void payTicked(){
        if(ticket>0){
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"正在卖>>"+ticket);
            ticket--;
        }
    }
}
3.静态同步方法
public class RunnableImpl implements Runnable {
    private static int ticket =100;//变量也必须是静态
    
    @Override
    public void run() {
        while(true){
             payTicked();
            }
    }
    public static synchronized void payTicked(){//静态方法
        if(ticket>0){//变量也必须是静态
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"正在卖>>"+ticket);
            ticket--;
        }
    }
}

三、同步技术原理

1.同步对象

使用一个锁对象,这个锁对象叫同步锁。也叫对象锁,也叫对象监视器。3个线程一起抢占cpu的执行权,谁墙到了谁执行run方法进行买票。t0抢到了cpu的执行权,执行run,遇到suynchronized代码块,会检查syunchronized代码块是否有锁对象,如果有,就会获取到锁对象,进入到同步中执行。

t1抢到了cpu的执行权,执行run,遇到suynchronized代码块,会检查syunchronized代码块是否有锁对象,法线没有(t0占用),t1就会进入阻塞状态。会一直等待t0线程释放锁对象,释放后,t1获取锁对象,继续执行。

**总结:**同步中的线程,没有执行完毕不会释放锁,同步外的线程进不去同步

程序会频繁的判断锁,效率会降低

在这里插入图片描述

2.同步方法

同步方法也会把方法内部的代码锁住。同步方法的锁对象就是调用方法的对象。是***this***。

3.静态同步方法

锁对象不能是this(static先与对象产生).是本类的class属性,class文件对象(反射)。

四、解决线程安全问题Lock锁

java.util.concurrent.locks   Lock 实现提供了比使用 synchronized 方法和语句可获得的更广泛的锁定操作。此实现允许更灵活的结构,可以具有差别很大的属性,可以支持多个相关的 Condition 对象。 
1.lock接口中的方法:

void lock():获取锁

void unlock():释放锁

2.实现类

java.util.concurrent.locks.javaReentrantLock implements Lock;

3.使用
public class RunnableImpl01 implements Runnable {
    private int ticket =100;
    Lock l = new ReentrantLock();
    @Override
    public void run() {
        while(true){
            l.lock();
            if(ticket>0){
                try {
                    Thread.sleep(1000);
                     System.out.println(Thread.currentThread().getName()+"正在卖>>"+ticket);
                ticket--;
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }finally{
                    l.unlock();
                }
            }
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值