线程安全问题举例

如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码。如果每次运行结果和 单线程运行的结果是一样的,而且其他的 变量的值也和预期的是一样的,就是线程安全的。
或者说:一个类或者程序所提供的接口对于线程来说是 原子操作或者多个线程之间的切换不会导致该接口的执行结果存在二义性,也就是说我们不用考虑同步的问题。
线程安全问题都是由 全局变量静态变量引起的。

若每个线程中对全局变量静态变量只有读操作,而无写操作,一般来说,这个全局变量是线程安全的;若有多个线程同时执行写操作,一般都需要考虑线程同步,否则的话就可能影响线程安全。

比如一个 ArrayList 类,在添加一个元素的时候,它可能会有两步来完成:1. 在 Items[Size] 的位置存放此元素;2. 增大 Size 的值。
单线程运行的情况下,如果 Size = 0,添加一个元素后,此元素在位置 0,而且 Size=1;
而如果是在 多线程情况下,比如有两个线程,线程 A 先将元素存放在位置 0。但是此时 CPU 调度线程A暂停,线程 B 得到运行的机会。线程B也向此 ArrayList 添加元素,因为此时 Size 仍然等于 0 (注意哦,我们假设的是添加一个元素是要两个步骤哦,而线程A仅仅完成了步骤1),所以线程B也将元素存放在位置0。然后线程A和线程B都继续运行,都增加 Size 的值。
那好,我们来看看 ArrayList 的情况,元素实际上只有一个,存放在位置 0,而 Size 却等于 2。这就是“线程不安全”了。
静态变量线程安全问题模拟:

Java代码   收藏代码
  1. /**  
  2.   * 线程安全问题模拟执行  
  3.   *  ------------------------------  
  4.   *       线程1      |    线程2  
  5.   *  ------------------------------  
  6.   *   static_i = 4;  | 等待  
  7.   *   static_i = 10; | 等待  
  8.   *    等待          | static_i = 4;  
  9.   *   static_i * 2;  | 等待  
  10.   *  ----------------------------- 
  11.  * */  
  12. public class Test implements Runnable  
  13. {  
  14.     private static int static_i;//静态变量   
  15.       
  16.     public void run()  
  17.     {  
  18.         static_i = 4;  
  19.         System.out.println("[" + Thread.currentThread().getName()  
  20.                 + "]获取static_i 的值:" + static_i);  
  21.         static_i = 10;  
  22.         System.out.println("[" + Thread.currentThread().getName()  
  23.                 + "]获取static_i*3的值:" + static_i * 2);  
  24.     }  
  25.       
  26.     public static void main(String[] args)  
  27.     {  
  28.         Test t = new Test();  
  29.         //启动尽量多的线程才能很容易的模拟问题   
  30.         for (int i = 0; i < 3000; i++)  
  31.         {  
  32.             //t可以换成new Test(),保证每个线程都在不同的对象中执行,结果一样   
  33.             new Thread(t, "线程" + i).start();  
  34.         }  
  35.     }  
  36. }  


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值