Java 多线程读者写者问题

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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 多线程的读者-问题是一个经典的同步问题,涉及到多个读线程和线程对共享数据的访问。以下是一个可能的Java线读者-问题的实现代码: ```java class ReaderWriter { int readers = 0; int writers = 0; int writeRequests = 0; public synchronized void startRead() throws InterruptedException { while (writers > 0 || writeRequests > 0) { wait(); } readers++; } public synchronized void endRead() { readers--; notifyAll(); } public synchronized void startWrite() throws InterruptedException { writeRequests++; while (readers > 0 || writers > 0) { wait(); } writeRequests--; writers++; } public synchronized void endWrite() { writers--; notifyAll(); } } class Reader implements Runnable { ReaderWriter readerWriter; public Reader(ReaderWriter readerWriter) { this.readerWriter = readerWriter; } public void run() { try { readerWriter.startRead(); // 读取共享数据的操作 Thread.sleep((int) (Math.random() * 1000)); readerWriter.endRead(); } catch (InterruptedException e) { e.printStackTrace(); } } } class Writer implements Runnable { ReaderWriter readerWriter; public Writer(ReaderWriter readerWriter) { this.readerWriter = readerWriter; } public void run() { try { readerWriter.startWrite(); // 入共享数据的操作 Thread.sleep((int) (Math.random() * 1000)); readerWriter.endWrite(); } catch (InterruptedException e) { e.printStackTrace(); } } } public class Main { public static void main(String[] args) { ReaderWriter readerWriter = new ReaderWriter(); for (int i = 0; i < 5; i++) { new Thread(new Reader(readerWriter)).start(); } for (int i = 0; i < 5; i++) { new Thread(new Writer(readerWriter)).start(); } } } ``` 在上面的代码中,`ReaderWriter`类是一个用于管理读者者访问共享数据的同步类。`startRead()`和`endRead()`方法用于读者访问共享数据的开始和结束,`startWrite()`和`endWrite()`方法用于者访问共享数据的开始和结束。在每个方法中使用`synchronized`关键字来保证同一时间只有一个线程可以执行。`notifyAll()`方法用于唤醒其他正在等待的线程。 `Reader`类和`Writer`类分别是读者线程的实现。在`run()`方法中,先调用对应的`startRead()`或`startWrite()`方法来获取访问权限,然后执行读取或入操作,最后调用对应的`endRead()`或`endWrite()`方法来释放访问权限。 在`main()`方法中创建了5个读者线程和5个线程,并启动它们。 以上代码是一种可能的多线读者-问题的实现,但并不是唯一的解决方案。在实际应用中,还需要考虑更多的场景和线程同步的细节,确保共享数据的一致性和线程的安全执行。 ### 回答2: Java线读者-问题可以通过使用synchronized关键字和wait()、notify()方法来实现。下面是一个简单的示例代码: ``` public class ReaderWriterProblem { private static final int MAX_READERS = 5; private static final int MAX_WRITERS = 2; private static int activeReaders = 0; private static boolean writerActive = false; public static void main(String[] args) { for (int i = 1; i <= MAX_READERS; i++) { new Thread(new Reader(i)).start(); } for (int i = 1; i <= MAX_WRITERS; i++) { new Thread(new Writer(i)).start(); } } static class Reader implements Runnable { private final int readerId; public Reader(int readerId) { this.readerId = readerId; } @Override public void run() { while (true) { try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } read(); } } private void read() { synchronized (ReaderWriterProblem.class) { while (writerActive) { // 如果有者在执行 try { ReaderWriterProblem.class.wait(); // 等待者完成 } catch (InterruptedException e) { e.printStackTrace(); } } activeReaders++; // 增加活跃读者数量 } // 执行读操作 System.out.println("Reader " + readerId + "正在执行读操作"); synchronized (ReaderWriterProblem.class) { activeReaders--; // 减少活跃读者数量 if (activeReaders == 0) { // 如果没有其他读者 ReaderWriterProblem.class.notifyAll(); // 唤醒所有等待的线程 } } } } static class Writer implements Runnable { private final int writerId; public Writer(int writerId) { this.writerId = writerId; } @Override public void run() { while (true) { try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } write(); } } private void write() { synchronized (ReaderWriterProblem.class) { while (activeReaders > 0 || writerActive) { // 如果有读者在执行或者有者在执行 try { ReaderWriterProblem.class.wait(); // 等待其他线程完成操作 } catch (InterruptedException e) { e.printStackTrace(); } } writerActive = true; } // 执行操作 System.out.println("Writer " + writerId + "正在执行操作"); synchronized (ReaderWriterProblem.class) { writerActive = false; ReaderWriterProblem.class.notifyAll(); // 唤醒所有等待的线程 } } } } ``` 在上述代码中,我们使用了一个整型变量`activeReaders`来记录当前活跃的读者数量,当读者开始读操作时,会先判断是否有者在执行,如果有,则等待者完成;然后增加`activeReaders`;接着执行读操作。读操作完成后,减少`activeReaders`,如果没有其他读者,则调用`notifyAll()`方法唤醒其他等待的线程。者也类似,当者开始操作时,会先判断是否有其他读者者在执行,如果有,则等待其他线程完成;然后执行操作;最后,设置`writerActive`为false,并调用`notifyAll()`方法唤醒其他等待的线程。 这种方式实现的读者-问题可以保证同一时间只能有一个者或多个读者执行操作,从而保证数据的一致性和并发访问的正确性。 ### 回答3: Java线读者-问题可以通过使用synchronized关键字和wait()、notify()方法实现。 首先,读者-问题是指多个读者线程可以同时读取数据,但线程只能在没有读者线程时才能入数据。 下面是使用Java实现的一个基本读者-问题的代码示例: ```java public class ReaderWriter { private int readers; private boolean isWriting; public ReaderWriter() { readers = 0; isWriting = false; } public synchronized void startRead() throws InterruptedException { while (isWriting) { wait(); } readers++; } public synchronized void endRead() { readers--; if (readers == 0) { notifyAll(); } } public synchronized void startWrite() throws InterruptedException { while (isWriting || readers > 0) { wait(); } isWriting = true; } public synchronized void endWrite() { isWriting = false; notifyAll(); } } ``` 上面的代码中,ReaderWriter类用于管理读者者的访问。其中,startRead()方法用于读者开始读取数据,endRead()方法用于读者结束读取;startWrite()方法用于者开始入数据,endWrite()方法用于者结束入。 在startRead()和startWrite()方法中,使用synchronized关键字修饰,确保同一时间只能有一个线程访问该方法。在读者读取数据时,如果有者在入,则调用wait()方法使读者线程进入等待状态。在入数据时,如果有其他读者在读取,也调用wait()方法使线程进入等待状态。 在endRead()和endWrite()方法中,对读者者的数量进行更新,并使用notifyAll()方法唤醒等待的线程。当所有读者都结束读取时,唤醒可能等待的线程;当者结束入时,唤醒可能等待的读者线程。 通过以上方法的实现,能够实现读者-问题的多线程并发访问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值