java-线程不安全会发生什么问题?

问题描述

线程不安全会发生什么问题?

线程不安全会发生什么问题?
线程不安全会发生什么问题?
线程不安全会发生什么问题?

解决方案

举例:
两个线程共享一个堆上的公共变量a
线程1和线程2都有如下代码
int c = a;
c = c + 1;
a = c; //实际上可能写的是a++,但是a++看上去是原子的,为了演示,写成如上的形式。
假设a=100,按照预期,两个线程各执行一次,结果应该是102,但是可能出现101:
线程1:c = a; c = c +1;,此时c = 101,线程2执行c = a,此时a还是100,c = c + 1,c= 101。然后线程1写回101,线程2也写回101。

解决方案二:

程序运行结果出错,死锁,等等

解决方案三:

关于死锁,举例:
线程1锁住a,等待b
在此过程中,如果线程2占用的是b,并且等待a,那么两个线程会无限等待下去,这就是姜昆小品中说的“工厂不给幼儿园开介绍信,幼儿园没有义务给工厂开介绍信;没有幼儿园的介绍信,工厂没有义务先给幼儿园开介绍信”

解决方案四:

有的时候会发生更糟糕的事情,比如说第一个程序,C++中一个线程释放了指针,另一个线程还在使用这个指针,那么有可能会破坏内存,导致不可预知的问题。

解决方案五:

最好自己写个测试,最常见的就是数据出错,

解决方案六:

有可能出现多个线程先后更改数据,从而造成所得到的数据是脏数据。

解决方案七:

不能按照你写的逻辑得到正确的结果!例如你要计数,如果你不是线程安全的,可能你获得计数结果就是错误的!

解决方案八:

多线程带来的风险有三种,安全性问题,活跃性问题,性能问题。
首先,安全性问题,《Java并发编程》一书中对类的线程安全的定义是:当多个线程访问某个类时,不管运行环境采用何种调度方式或者这些线程将如何交替执行,并且在主调用代码中不需要任何额外的同步或者协同,这个类都能表现正确的行为,那么这个类就是线程安全的。
相反这段描述的另一面就是线程不安全的,即可能出现某个类的行为与其规范的不一致。例如:数值序列生成器,简单的共享变量++操作。

 public class UnsafeSequence{
     private int value;
         public int getNext(){
             return value++;
        }
 }

这段代码的getNext如果执行时机不对,那么两个线程在调用getNext时会得到相同的值,因为value++操作本身包含三种独立的操作,读value,value+1,写入value,这三个动作如果在多个线程之间交替执行,就可能导致最终的数据不合常理。也就是说这段代码存在安全性问题。
其次,活跃性问题是指当某个操作无法继续执行下去时就会发生活跃性问题,如锁机制下线程死锁的问题。
最后,性能问题,多线程程序中,线程的调度,频繁的上下文切换等操作会带来极大的开销,也是编写多线程应用需要权衡的问题。

时间: 2024-08-30 14:31:36

java-线程不安全会发生什么问题?的相关文章

如何中断JAVA线程

如何中断JAVA线程 程序是很简易的.然而,在编程人员面前,多线程呈现出了一组新的难题,如果没有被恰当的解决,将导致意外的行为以及细微的.难以发现的错误.       在本篇文章中,我们针对这些难题之一:如何中断一个正在运行的线程.                                                                                      背景     中断(Interrupt)一个线程意味着在该线程完成任务之前停止其正在进行的一切,

Java线程模型缺陷研究

Java 编程语言的线程模型可能是此语言中最薄弱的部分.它完全不适合实际复杂程序的要求,而且也完全不是面向对象的.本文建议对 Java 语言进行重大修改和补充,以解决这些问题. Java 语言的线程模型是此语言的一个最难另人满意的部分.尽管 Java 语言本身就支持线程编程是件好事,但是它对线程的语法和类包的支持太少,只能适用于极小型的应用环境. 关于 Java 线程编程的大多数书籍都长篇累牍地指出了 Java 线程模型的缺陷,并提供了解决这些问题的急救包(Band-Aid/邦迪创可贴)类库.我

Java线程的深入探讨

一般来说,我们把正在计算机中执行的程序叫做"进程"(Process) ,而不将其 称为程序(Program).所谓"线程"(Thread),是"进程"中某个单一顺序的控制流. 新兴的操作系统,如Mac,Windows NT,Windows 95等,大多采用多线程的概念,把线 程视为基本执行单位.线程也是Java中的相当重要的组成部分之一. 甚至最简单的Applet也是由多个线程来完成的.在Java中,任何一个Applet的 paint()和upd

Java线程机制(五) 等待与通知机制

在之前我们关于停止Thread的讨论中,曾经使用过设定标记done的做法,一旦done设置为true,线程就会 结束,一旦为false,线程就会永远运行下去.这样做法会消耗掉许多CPU循环,是一种对内存不友好的行为. java中的对象不仅拥有锁,而且它们本身就可以通过调用相关方法使自己成为等待者和通知者. Object对象本身有两个方法:wait()和notify().wait()会等待条件的发生,而notify()会通知正在 等待的线程此条件已经发生,它们都必须从synchronized方法或

Java线程机制(三) synchronized和volatile的使用

现在开始进入线程编程中最重要的话题---数据同步,它是线程编程的核心,也是难点,就算我们理解了 数据同步的基本原理,但是我们也无法保证能够写出正确的同步代码,但基本原理是必须掌握的. 要 想理解数据同步的基本原理,首先就要明白,为什么我们要数据同步? public class CharacterDisplayCanvas extends JComponent implements CharacterListener { protected FontMetrics fm; protected ch

线程基础(第二部分)Java线程的缺陷和副作用几解决办法

<--在线程基础的第二部分中,我们将了解一下使用Java线程的缺陷和副作用,以及在SUN JDK 1.2中是如何修改线程的运行机制的--> 在上篇文章<Java 101之线程基础>中,我们介绍了线程的概念以及如何使用线程.这个月,我们将转到更高级的话题,包括线程的缺陷及副作用,以及在SUN JDK 1.2中,是如何改进线程的运行机制的. synchronize(同步) 让我们回忆一下上篇文章中讲的:线程允许两个或者更多个进程同时执行.实际上,这些线程也可以共享对象和数据,在这种情形

Java线程的讨论与应用

一.为什么要研究和使用线程 一般来说,计算机正在执行的程序称作进程(process),进程有不同的地址空间并且是在同一系统上运行的不同程序,如WORD和Excel,进程间的通讯是很费时而且有限的.上下文切换.改变运行的进程也是非常复杂的.进程间通讯复杂,可能需要管道.消息队列.共享内存(sharedmemory)或信号处理来保证进程间的通讯.尽管许多程序都在运行,但一次只能与一个程序打交道. 线程(thread)是指进程中单一顺序的控制流.又称为轻量级进程.线程则共享相同的地址空间并共同构成一个

Java线程:并发协作-死锁

发生死锁的原因一般是两个对象的锁相互等待造成的. 在<Java线程:线程的同步与锁>一文中,简述死锁的概念与简单例子,但是所给的例子是不完整的,这里给出一个完整的例子. /** * Java线程:并发协作-死锁 * * @author Administrator */ public class Test { public static void main(String[] args) { DeadlockRisk dead = new DeadlockRisk(); MyThread t1 =

基于JVMTI实现Java线程的监控

随着多核 CPU 的日益普及,越来越多的 Java 应用程序使用多线程并行计算来充分发挥整个系统的性能.多线程的使用也给应用程序开发人员带来了巨大的挑战,不正确地使用多线程可能造成线程死锁或资源竞争,导致系统瘫痪.因此,需要一种运行时线程监控工具来帮助开发人员诊断和跟踪 Java 线程状态的切换.JDK 1.5 及其后续版本提供了监控虚拟机运行状态的接口 JVMTI. JVMTI 工具接口 随着多核 CPU 技术的发展,多线程编程技术被广泛地应用,从而充分发挥整个系统的性能.Java 语言对多线