java线程系列文章之一(线程的安全性)

本文来自:高爽|Coder,原文地址:http://blog.csdn.net/ghsau/article/details/7421217,转载请注明。      

当我们查看JDK API的时候,总会发现一些类说明写着,线程安全或者线程不安全,比如说StringBuilder中,有这么一句,“将StringBuilder 的实例用于多个线程是不安全的。如果需要这样的同步,则建议使用StringBuffer。 ”,那么下面手动创建一个线程不安全的类,然后在多线程中使用这个类,看看有什么效果。

  

1 public class Count {
2     private int num;
3     public void count() {
4         for(int i = 1; i <= 10; i++) {
5             num += i;
6         }
7         System.out.println(Thread.currentThread().getName() + "-" + num);
8     }
9 }

在这个类中的count方法是计算1一直加到10的和,并输出当前线程名和总和,我们期望的是每个线程都会输出55。 

ThreadTest.java:

 1 public class ThreadTest {
 2     public static void main(String[] args) {
 3         Runnable runnable = new Runnable() {
 4             Count count = new Count();
 5             public void run() {
 6                 count.count();
 7             }
 8         };
 9         for(int i = 0; i < 10; i++) {
10             new Thread(runnable).start();
11         }
12     }
13 }

这里启动了10个线程,看一下输出结果:

 1 Thread-0-55
 2 Thread-1-110
 3 Thread-2-165
 4 Thread-4-220
 5 Thread-5-275
 6 Thread-6-330
 7 Thread-3-385
 8 Thread-7-440
 9 Thread-8-495
10 Thread-9-550 

只有Thread-0线程输出的结果是我们期望的,而输出的是每次都累加的,这里累加的原因以后的博文会说明,那么要想得到我们期望的结果,有几种解决方案:

1. 将Count中num变成count方法的局部变量;

1 public class Count {
2     public void count() {
3         int num = 0;
4         for(int i = 1; i <= 10; i++) {
5             num += i;
6         }
7         System.out.println(Thread.currentThread().getName() + "-" + num);
8     }
9 }

 2. 将线程类成员变量拿到run方法中,这时count引用是线程内的局部变量;

 1 public class ThreadTest4 {
 2     public static void main(String[] args) {
 3         Runnable runnable = new Runnable() {
 4             public void run() {
 5                 Count count = new Count();
 6                 count.count();
 7             }
 8         };
 9         for(int i = 0; i < 10; i++) {
10             new Thread(runnable).start();
11         }
12     }
13 }&nbsp;

 
        上述测试,我们发现,存在成员变量的类用于多线程时是不安全的,不安全体现在这个成员变量可能发生非原子性的操作,而变量定义在方法内也就是局部变量是线程安全的。想想在使用struts1时,不推荐创建成员变量,因为action是单例的,如果创建了成员变量,就会存在线程不安全的隐患,而struts2是每一次请求都会创建一个action,就不用考虑线程安全的问题。所以,日常开发中,通常需要考虑成员变量或者说全局变量在多线程环境下,是否会引发一些问题。

原文地址:https://www.cnblogs.com/lintong/p/4371329.html