码迷,mamicode.com
首页 > 其他好文 > 详细

读者写者模式举例

时间:2015-08-11 10:23:27      阅读:99      评论:0      收藏:0      [点我收藏+]

标签:

/* 
 * 读者写者问题 
 *  
 * */
class Buffer
{
    public String buffer[] = new String[10];//用于标识共享的缓冲区  
    public boolean flag = false;//用于标识缓冲区的状态  
    public int i=0;//表示向缓冲区写的位置   
}
class Writer implements Runnable //可能会有多个读  
{
    private Buffer buffer;
    public Writer(Buffer buffer)
    {
        this.buffer = buffer;
    }
    @Override
    public void run()
    {
        while (true)//用循环表示不停地向里面写东西  
        {
            synchronized (buffer)//这里面是同步代码块,里面都是对共享数据的操作,只能有一个线程操作  
            {
                while (buffer.flag)//如果共享数据flag=true,表示共享数据块里面已经写入数据还没有被读出来,如果再写的话,就要等待,这也是控制读一次写一次的关键  
                {
                    try
                    {
                        buffer.wait();
                    } catch (InterruptedException ex)
                    {
                        System.out.println(ex.getMessage());
                        ex.printStackTrace(System.out);
                    }
                }
                if (buffer.i == buffer.buffer.length)
                {
                    System.out.println("缓冲区已满,清空后再写");
                    buffer.buffer = new String[10];
                    buffer.i = 0;
                }
                buffer.buffer[buffer.i] = "字符串" + buffer.i;//这里面是向缓冲区里面写入数据的操作  
                System.out.println(Thread.currentThread().getName()+"向" + buffer.i + "写入:" + buffer.buffer[buffer.i]);
                buffer.i = buffer.i + 1;//防止数据的溢出  
                buffer.flag = true;
                buffer.notifyAll();
            }
            try
            {
                Thread.sleep(100);
            } catch (Exception ex)
            {
                System.out.println(ex.getMessage());
                ex.printStackTrace(System.out);
            }
        }
    }
}
class Reader implements Runnable
{
    private Buffer buffer;
    public Reader(Buffer buffer)
    {
        this.buffer = buffer;
    }
    @Override
    public void run()
    {
        while (true) //表示 读者不停在进行读操作  
        {
            synchronized (buffer)
            {
                while (!buffer.flag)
                {
                    try
                    {
                        buffer.wait();
                    } catch (InterruptedException ex)
                    {
                        System.out.println(ex.getMessage());
                        ex.printStackTrace(System.out);
                    }
                }
                System.out.print(Thread.currentThread().getName()+" 读取:");
                for (int i = 0; i < buffer.i; i++)
                {
                    System.out.print(buffer.buffer[i]);
                }
                System.out.println();
                buffer.flag = false;
                buffer.notifyAll();
            }
            try
            {
                Thread.sleep(100);//这句话没有什么意思,只是为了让输出会慢一点,在这个程序里面没有其他作用,加不加都一个样  
            } catch (InterruptedException ex)
            {
                System.out.println(ex.getMessage());
                ex.printStackTrace(System.out);
            }  
        }
    }
}
/* 
 *              上面的程序在只有一个读者和一个写者的情况 下没有任何错误,写者没写进一个数据,读者就读取数据一次 
 *        而在实际中,线程的个数可能会有很多种,在有多个读者和多个写者的情况下,上面的代码又会现在一个新的问题--- 
 *              那就是在写一个而读多次,或者是读一次而写多次,这将会出现安全性问题,下面我们讨论一下为什么出现这种问题 
 *        假设现在有两个读者和两个写者,现在执行的线程是读者,两个写者wait 中,读者执行完后唤醒其中的一个写者,然后写者开始执行 
 *          当写者执行完成后,下面又开始要唤醒线程,因为上面的两个读线程和一个写线程都在等待当中,如果唤醒的是再是写线程,因为里面用的是 
 *          if(buffer.flag) 这个在wait 之前就已经判断过了,所以会直接执行下面的语句,那么就是在读了一次的情况下,而连接写了两次,如果写的线程更多 的话, 
 *          那么就会出现更多 的问题,所以上面的if(buffer.flag)要改成while(buufer.flag) buffer.notify 要改成buffer.notifyAll();这样就正确了       
 *       
 *      上面的程序由于由while(flag) notifyAll  控制,所以,无论 写者线程或读者线程有多少个,无论是否相同,它们都将遵守读一个写一个的顺序了 
 *                       
 */
public class ReaderandWriter
{
    public static void main(String args[])
    {
        Buffer buffer = new Buffer();
        Thread t1 = new Thread(new Writer(buffer));
        Thread t2 = new Thread(new Writer(buffer));
        Thread t3 = new Thread(new Reader(buffer));
        Thread t4 = new Thread(new Reader(buffer));
        t1.start();
        t2.start();  
        t3.start();
        t4.start();
    }
}

读者写者模式举例

标签:

原文地址:http://my.oschina.net/liuyuanyuangogo/blog/490504

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!