为什么会出现这样的线程安全问题

问题描述

/*]在进行flag标志判断的时候用的是if,而不是while,如果用while就不会有安全问题,用了,c1,c2两个监视器,按理每次都只唤醒一个对方线程,为什么还会出现安全问题呢*/importjava.util.concurrent.locks.*;classResource{privateStringname;privateintcount=1;privatebooleanflag=false;privateLocklock=newReentrantLock();privateConditionc1=lock.newCondition();privateConditionc2=lock.newCondition();publicvoidset(Stringname){lock.lock();try{if(flag){try{c1.await();}catch(InterruptedExceptione){}}this.name=name+count++;System.out.println(Thread.currentThread().getName()+"....生产者..."+this.name);flag=true;c2.signal();}finally{lock.unlock();}}publicvoidout(){lock.lock();try{if(!flag){try{c2.await();}catch(InterruptedExceptione){}}System.out.println(Thread.currentThread().getName()+"....消费者......."+this.name);flag=false;c1.signal();}finally{lock.unlock();}}}classProducerimplementsRunnable{Resourcer;Producer(Resourcer){this.r=r;}publicvoidrun(){while(true)r.set("烤鸭");}}classConsumerimplementsRunnable{Resourcer;Consumer(Resourcer){this.r=r;}publicvoidrun(){while(true)r.out();}}classProducerConsumerDemo2{publicstaticvoidmain(String[]args){Resourcer=newResource();Producerp=newProducer(r);Consumerc=newConsumer(r);Threadt1=newThread(p);Threadt2=newThread(p);Threadt3=newThread(c);Threadt4=newThread(c);t1.start();t2.start();t3.start();t4.start();}}

解决方案

解决方案二:
楼主要学会贴代码publicclassDemo{publicstaticvoidmain(String[]args){if(args.length==0){System.out.println("noargs.");}}}

像楼主的代码层级多,又没有排版,没法看。
解决方案三:
比如说这样的情形:第一次生产者放东西,OK没问题,第二次生产者放东西,要wait了。现在另一个生产者和二个消费者在竞争,一个消费者得到了锁,执行后让那个等待的生产者复活。现在4个人重新竞争。其中一个生产者(没有wait的那个)得到了锁,执行了一次。这时原则上应该让消费者来了。然而另一个生产者(wait过的那个)又得到了锁,成功执行了一次(从wait()后面开始执行),不进行判断这不安全。------------------------------------------------------------------------------------------------------------有一点不太确定,刚唤醒的线程,不知道是不是比那些没有等待过的线程优先级要高点。

时间: 2024-08-02 17:26:50

为什么会出现这样的线程安全问题的相关文章

Spring-利用ThreadLocal解决线程安全问题

ThreadLocal是什么 ThreadLocal,顾名思义,它不是一个线程,而是线程的一个本地化对象.当工作于多线程中的对象使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程分配一个独立的变量副本.所以每一个线程都可以独立地改变自己的副本,而不会影响其他线程所对应的副本.从线程的角度看,这个变量就像是线程的本地变量,这也是类名中"Local"所要表达的意思. ThreadLocal的方法很简单,主要的就是4个方法 1234567891011 // 设

Servlet和JSP的线程安全问题

js|servlet|安全|问题 编写Servlet和JSP的时候,线程安全问题很容易被忽略,如果忽视了这个问题,你的程序就存在潜在的隐患. 1.Servlet的生命周期 Servlet的生命周期是由Web容器负责的,当客户端第一次请求Servlet时,容器负责初始化Servlet,也就是实例化这个Servlet类.以后这个实例就负责客户端的请求,一般不会再实例化其他Servlet类,也就是有多个线程在使用这个实例.Servlet之所以比CGI效率高就是因为Servlet是多线程的.如果该Ser

UNIX编程中错误输出的线程安全问题

系统调用失败原因分析 在 UNIX 编程中,我们会经常使用系统调用来完成期望的功能:而与此同时,我们也需要付出大段 的代码来检测.输出错误和其他意外情况. 以下是系统调用失败的可能原因: 系统可能出现资源短缺或者程序使用的资源可能超过系统为单个程序规定的上限.常见的情况有: 程序可能尝试分配大量内存,或者同时打开很多文件等. 程序执行操作时,可能会由于权限不足而被系统阻止.例如,程序可能会试图写一个只读的文件, 或者企图访问其他进程的内存空间. 传入系统调用的参数可能无效,原因可能是用户提供无效

HttpSession的线程安全问题及注意事项

HttpSession session = request.getSession(); List<Product> list = session.getAttribute("productCart"); myService.save(list); // 保存购物车数据到数据库 这个对象会被多次使用,也会被同一个用户的多个页面使用,所以他对于系统来说是线程不安全的. 比如用户在从产品列表里面选择产品,这面选择3种,他点了查看购物车 该用户还开了另一个页面,继续选择产品. 此时

在多线程中使用静态方法是否有线程安全问题

   类的成员分为两类,静态成员(static member)和实例成员(instance member).静态成员属于类,实例成员则属于对象,即类的实例.     简单讨论一下在一个类中使用静态字段(static field)和静态方法(static method)是否会有线程安全问题.      我们在知道, 静态字段(static field)和静态方法(static method)的调用是通过类来调用.静态方法不对特定的实例操作,只能访问静态成员.实例方法可对特定的实例操作,既能访问静态

java-Java 单例线程安全问题

问题描述 Java 单例线程安全问题 public class A { public final static A INSTANCE = new A(); private A(){} }这个类是不是线程安全的呢?单例能这样写吗? 解决方案 这种写法是线程安全的.但是在该类一开始被加载的时候INSTANCE = new A()就会被执行.具体参考陈皓的博客深入浅出单实例Singleton设计模式介绍得非常详细. 解决方案二: 如果单线程中初始化,多线程中应用就没问题. 解决方案三: 深入解析单例线

java初学者求教:关于线程安全问题

问题描述 java初学者求教:关于线程安全问题 List list = new ArrayList(); list = Collections.synchronizedList(list); 这时list是线程安全的,那么当我用增强for循环遍历list, 并且使用list的remove方法时,是不是就不安全了? 另外,如果list集合中存放的是一些带有图片的对象,比如dog public void paintTest(Graphisc g){ for(dog d:list){ g.drawIm

多线程-高手请进!!!---线程安全问题,怎么解决new String 问题

问题描述 高手请进!!!---线程安全问题,怎么解决new String 问题 场景是:一个订单号只能一个在付款,只能一个线程处理,不同的订单号支持并发处理 现在如果是new String("20140719140818");就有问题怎么解决 如果不是new出来的,什么情况会出现问题 public class Test { public static void main(String[] args) { new Thread(){ public void run(){ pay(&quo

单例模式与线程安全问题浅析

           最近看到到Struts1与Struts2的比较,说Struts1的控制器是单例的,线程不安全的:Struts2的多例的,不存在线程不安全的问题.之后又想到了之前自己用过的HttpHandler...这些类,好像单例的线程安全问题确实是随处可见的.但是只是知道这个是不安全的,也没有认真分析过.接下来就仔细分析下. 一,修改单例模式代码       首先我先写一段单例类的代码:          /** * @ClassName: Sigleton * @Description

多线程-关于Vector的线程安全问题

问题描述 关于Vector的线程安全问题 一个线程做删除该集合元素的同时,其他线程做查询工作,会有安全问题吗 解决方案 当然会有安全问题,比如说另一个线程持有一个迭代器对象,那么会导致迭代器状态无效.你有两个办法,一个是锁住向量变量,一个是查询的时候先复制一个vector的副本.关键看你对同步的要求和为读还是写优化(程序里查询的多还是修改的多) 解决方案二: Java的vector的所有单个方法都是线程安全的,但是如果存在复合操作例如先判断如果不存在就put那么实际上你这两个动作前后是有依赖关系