专栏首页desperate633共享资源的线程安全性Local VariablesLocal Object ReferencesObject Member VariablesThe Thread Control Escape Rul

共享资源的线程安全性Local VariablesLocal Object ReferencesObject Member VariablesThe Thread Control Escape Rul

  • 局部变量
  • 局部对象变量的引用
  • 对象成员变量
  • 线程控制权原则

如果某段代码可以正确的被多线程并发的执行,那么我们就称这段代码是线程安全的,如果一段代码是线程安全的那么他肯定不会出现资源竞速的问题。资源竞速的问题只发生在多个线程同时更新共享资源的情况。因此,我们我们需要很清楚的知道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

这样就出现了问题。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Java线程池的工作原理,好处和注意事项

    、 一个线程池管理了一组工作线程, 同时它还包括了一个用于放置等待执行 任务的任务队列(阻塞队列) 。

    desperate633
  • Java并发之嵌套管程锁死(Nested Monitor Lockout)具体的嵌套管程死锁的例子Nested Monitor Lockout vs. Deadlock

    可以看到,lock()方法首先在”this”上同步,然后在monitorObject上同步。如果isLocked等于false,因为线程不会继续调用monito...

    desperate633
  • Java并发之“饥饿”和“公平锁”(Starvation and Fairness)java中发生线程饥饿的原因java中实现公平锁公平锁性能考虑

    如果一个线程的cpu执行时间都被其他线程抢占了,导致得不到cpu执行,这种情况就叫做“饥饿”,这个线程就会出现饥饿致死的现象,因为永远无法得到cpu的执行。解决...

    desperate633
  • Android 面试必备 - 线程

    java thread的运行周期中, 有几种状态, 在 java.lang.Thread.State 中有详细定义和说明:

    用户2965908
  • Java学习之多线程

    创建Thread类的子类时,首先声明子类的构造方法,其次用定义的run()方法覆盖Thread类的run()方法,即将自己要执行的程序区块写入run()方法中。

    cherishspring
  • 什么是线程安全,以及并发需要知道的几个概念

    众所周知,在Java的知识体系中,并发编程是非常重要的一环,也是面试的必问题,一个好的Java程序员是必须对并发编程这块有所了解的。为了追求成为一个好的Java...

    Debian社区
  • 并发编程

    线程安全 线程安全概念 : 当多个线程访问某一个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法)就是线程安全的. ...

    海仔
  • 【转】Java并发的AQS原理详解

    每一个 Java 的高级程序员在体验过多线程程序开发之后,都需要问自己一个问题,Java 内置的锁是如何实现的?最常用的最简单的锁要数 ReentrantL...

    一枝花算不算浪漫
  • 打通 Java 任督二脉 —— 并发数据结构的基石

    每一个 Java 的高级程序员在体验过多线程程序开发之后,都需要问自己一个问题,Java 内置的锁是如何实现的?最常用的最简单的锁要数 ReentrantLoc...

    老钱
  • Java多线程Thread VS Runnable详解

    进程是程序在处理机中的一次运行。一个进程既包括其所要执行的指令,也包括了执行指令所需的系统资源,不同进程所占用的系统资源相对独立。所以进程是重量级的任务,它们之...

    Java后端工程师

扫码关注云+社区

领取腾讯云代金券