线程安全和同步问题

线程安全和同步问题:

使用线程同步解决线程安全问题:
当多个线程访问一个对象时,如果不用考虑这些线程在运行时环境下的调度和交替执行,也不需要进行额外的同步,或者在调用方进行任何其他的协调操作,调用这个对象的行为都可以获得正确的结果,那这个对象就是线程安全的,否则就是不安全的。

1,解决线程安全问题的一种方案:使用同步代码块
格式:sychronized(锁对象)
{
可能出现的线程安全问题的代码(访问了共享数据的代码)
}

例如:

 Object obj = new Object();
      private int ticket = 100;

      @Override
      public void run() {
          while (true) {
              synchronized (obj) {
                  if (ticket > 0) {
                      //提高安全问题出现的频率
                      try {
                          Thread.sleep(20);
                      } catch (InterruptedException e) {
                          e.printStackTrace();

                          System.out.println(Thread.currentThread().getName() + "售出" + ticket--);
                      }
                  }
              }

          }
      }

2,同步代码块保证了线程的安全问题,但是进程在执行同步代码块的时候每次都要判断锁的状态,非常消耗资源,效率比较低。

3,第二种方案:同步方法来解决。同步方法的对象是当前创建的对象(this)在这里插入图片描述
例如:


      private int ticket = 100;

      @Override
      public void run() {
          while (true) {
              payticket();
          }
      }

      public synchronized  void payticket()
      {
          if (ticket > 0) {
              //提高安全问题出现的频率
              try {
                  Thread.sleep(20);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
                  System.out.println(Thread.currentThread().getName() + "售出" + ticket--);
              }
          }

3,第三种方案:静态同步方法:锁对象是本类所在的所有对象:注意的是,静态方法只能访问静态变量。描述的对象为:(类.class)

举例:


      private static int ticket = 100;

      @Override
      public void run() {
          while (true) {
              payticketstatic();
          }
      }
      public  static synchronized  void payticketstatic()
      {
          if (ticket > 0) {
              //提高安全问题出现的频率
              try {
                  Thread.sleep(20);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
                  System.out.println(Thread.currentThread().getName() + "售出" + ticket--);
              }
          }

4,解决线程安全的方案四:lock锁;(上锁:lock() )(释放锁 (unlock() )),注意的是,这里要导入的类有两个
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

在这里插入图片描述
在这里插入图片描述

举例:


   //创建锁对象
      Lock l = new ReentrantLock();

      private int ticket = 100;

      @Override
      public void run() {
          while (true) {

              //上锁
              l.lock();
              if (ticket > 0) {
                  //提高安全问题出现的频率
                  try {
                      Thread.sleep(20);
                      System.out.println(Thread.currentThread().getName() + "售出" + ticket--);
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
                  //这里加上finally,表示无论程序是否异常,都会释放锁,提高程序效率
                  finally {
                      //释放锁
                      l.unlock();
                  }

                  }

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值