共享资源的线程安全性

作者: 六尺帐篷 | 来源:发表于2017-06-10 15:36 被阅读95次
  • 局部变量
  • 局部对象变量的引用
  • 对象成员变量
  • 线程控制权原则

如果某段代码可以正确的被多线程并发的执行,那么我们就称这段代码是线程安全的,如果一段代码是线程安全的那么他肯定不会出现资源竞速的问题。资源竞速的问题只发生在多个线程同时更新共享资源的情况。因此,我们我们需要很清楚的知道java线程在执行的时候共享了哪些资源?什么类型的资源?不同类型的资源会引起不同的问题。

Local Variables

局部变量存储在各自的线程栈中,这就意味着局部变量不会被不同线程共享,每个线程都是私有的,所以局部的原始数据类型的变量是线程安全的。

public void someMethod(){

  long threadSafeInt = 0;

  threadSafeInt++;
}

Local Object References

局部的引用变量相对有点不同,我们知道引用本身都是私有的,但是对象的引用确实共享的,因为对象不存在线程的本地栈中,而是存在共享堆中。

如果一个对象在局部创建,而且从来没有离开创建他所在的那个方法,那么就是线程安全的,进一步的,我们也可以把这个引用变量传给其他方法和对象,只要这些对象和方法对其他的线程不可用。

public void someMethod(){

  LocalObject localObject = new LocalObject();

  localObject.callMethod();
  method2(localObject);
}

public void method2(LocalObject localObject){
  localObject.setValue("value");
}

Object Member Variables

我们都知道成员变量是存储在堆上的,因此如果两个线程调用同一个对象的方法,这个方法更新了对象成员变量的话,这个方法就不是线程安全的。看下面这个例子就不是线程安全的:

public class NotThreadSafe{
    StringBuilder builder = new StringBuilder();

    public add(String text){
        this.builder.append(text);
    }
}
NotThreadSafe sharedInstance = new NotThreadSafe();

new Thread(new MyRunnable(sharedInstance)).start();
new Thread(new MyRunnable(sharedInstance)).start();

public class MyRunnable implements Runnable{
  NotThreadSafe instance = null;

  public MyRunnable(NotThreadSafe instance){
    this.instance = instance;
  }

  public void run(){
    this.instance.add("some text");
  }
}

如果两个线程操作的是不同的对象实例,那么实际上又是线程安全的。

new Thread(new MyRunnable(new NotThreadSafe())).start();
new Thread(new MyRunnable(new NotThreadSafe())).start();

所以,只要我们控制好对对象的访问和操作,即使对象不是线程安全的,我们也可以写出线程安全的代码。

The Thread Control Escape Rule

我们可以运用** The Thread Control Escape Rule **来判断一个线程是否是线程安全的。

If a resource is created, used and disposed within
the control of the same thread,
and never escapes the control of this thread,
the use of that resource is thread safe.
如果一个资源的创建,使用,操作都被同一个线程所控制,而且从未逃脱过这个线程的控制,那么对于这个资源的使用就是线程安全的。

即使一个对象是线程安全的,如果这个对象指向一个文件或者数据库的话,你的应用程序可能不会是线程安全的,例如,如果线程1和线程2各自创造他们自己的数据库连接,那么这两个连接是相互独立的,但是他们对于数据库的访问由于是访问同一个数据库资源,那么就可能会出现线程安全的问题。

check if record X exists
if not, insert record X

假设出现这样的执行顺序

Thread 1 checks if record X exists. Result = no
Thread 2 checks if record X exists. Result = no
Thread 1 inserts record X
Thread 2 inserts record X

这样就出现了问题。

相关文章

  • 共享资源的线程安全性

    局部变量 局部对象变量的引用 对象成员变量 线程控制权原则 如果某段代码可以正确的被多线程并发的执行,那么我们就称...

  • 4.04.并发问题

    针对线程的安全性问题,我们需要使用同步(就是要加锁,共享资源只能一个人同时访问)锁语法: 代码被多个线程访问 代码...

  • java 并发编程相关知识点(一)

    共享资源 什么是共享资源?共享资源指的是多个线程同时对同一份资源进行访问(读写操作),被多个线程访问的资源...

  • redis构建应用组件-分布式锁

    分布式锁的作用或者说意义在于 什么是锁 对于单进程应用,有时需要保证对共享资源读取和修改的线程安全性,Java提供...

  • 线程之间如何进行通讯

    什么是多线程之间的通讯 多个线程对同一个资源(共享资源),每个线程对共享资源做的动作或者操作不同 多线程通讯的生产...

  • Java Semaphore源码分析

    Semaphore介绍 Semaphore,信号量,用来控制访问特定数量的共享资源,当线程线程想访问共享资源时,必...

  • ios线程安全

    1.单线程,任务依次串行执行是不存在线程安全问题的。2.多线程访问共享资源而不去修改共享资源也可以保证线程安全 苹...

  • java实现多线程的两种方式

    Java需要并发控制三个原因: 多线程环境 存在共享资源 多个线程操作(修改)共享资源 下面分别用继承Thread...

  • AQS原理解析

    AQS核心思想 如果被请求的共享资源空闲,则将当前请求线程设置为有效的工作线程,并将共享资源设置为锁定状态。如果被...

  • AQS是什么

    AQS的核心思想是,如果被请求的共享资源空闲,则将当前请求资源的线程设置为有效的工作线程,并将共享资源设置为锁定状...

网友评论

本文标题:共享资源的线程安全性

本文链接:https://www.haomeiwen.com/subject/yqzqqxtx.html