zoukankan      html  css  js  c++  java
  • Java线程(一):线程安全与不安全

     作为一个Java Web开发人员,很少也不需要去处理线程,因为服务器已经帮我们处理好了。记得大一刚学Java的时候,老师带着我们做了一个局域网聊天室,用到了AWT、Socket、多线程、I/O,编写的客户端和服务器,当时做出来很兴奋,回学校给同学们演示,感觉自己好NB,呵呵,扯远了。上次在百度开发者大会上看到一个提示语,自己写的代码,6个月不看也是别人的代码,自己学的知识也同样如此,学完的知识如果不使用或者不常常回顾,那么还不是自己的知识。大学零零散散搞了不到四年的Java,我相信很多人都跟我一样,JavaSE基础没打牢,就急忙忙、兴冲冲的搞JavaEE了,然后学习一下前台开发(html、css、JavaScript),有可能还搞搞jQuery、extjs,再然后是Struts、hibernatespring,然后听说找工作得会linux、oracle,又去学,在这个过程中,是否迷失了,虽然学习面很广,但就像《神雕侠侣》中黄药师评价杨过,博而不精、杂而不纯,这一串下来,感觉做Java开发好难,并不是学着难,而是知识面太广了,又要精通这个,又要精通那个,这只是我迷茫时候的想法,现在我已经找到方向了。

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

            Count.java:

    [java] view plain copy
     
     print?在CODE上查看代码片派生到我的代码片
    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:

    [java] view plain copy
     
     print?在CODE上查看代码片派生到我的代码片
    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个线程,看一下输出结果:

    [java] view plain copy
     
     print?在CODE上查看代码片派生到我的代码片
    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方法的局部变量;

    [java] view plain copy
     
     print?在CODE上查看代码片派生到我的代码片
    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引用是线程内的局部变量;

    [java] view plain copy
     
     print?在CODE上查看代码片派生到我的代码片
    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. }   

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

    本文来自:高爽|Coder,原文地址:http://blog.csdn.net/ghsau/article/details/7421217

    汇总:http://blog.csdn.net/ghsau/article/details/17609747

    看云:http://www.kancloud.cn/digest/java-thread/107455

  • 相关阅读:
    梯度下降法
    超平面
    感知机模型
    三角不等式
    统计学习方法基本概念
    Kaggle 的注册和使用
    win10 部署 Anaconda
    全概率和贝叶斯公式
    行列式
    伴随矩阵
  • 原文地址:https://www.cnblogs.com/csniper/p/5544446.html
Copyright © 2011-2022 走看看