`
liwanchun_xd
  • 浏览: 123452 次
  • 来自: ...
文章分类
社区版块
存档分类
最新评论

多线程读写同步

阅读更多
写模型

读写模型是一个稍微复杂一些的模型。
一份共享资源允许多个读者同时读取。但是只要有一个写者在写这份共享资源,任何其他的读者和写者都不能访问这份共享资源。
读写模型实现起来,不仅需要信号量机制,还需要额外的读者计数和写者计数。
public static final Object signal = new Object();
public static int readers = 0;
public static int writers = 0;

// 读者代码
… read() {

for(… ) { // 循环执行

synchronized(signal){
while( writers > 0 )
signal.wait(); // 如果有人在写,那么就放弃执行,进入待召队列

// 能够到达这里,说明没有人在写

readers ++ ; // 增加一个读者计数,表示本线程在读取
} // 这里出了synchronized范围,释放同步锁.以便其他线程读取.


// 进行一些读取操作

synchronized(signal){
readers --; // 读取完成,减少一个读者计数,表示本线程不在读取

signal.notifyAll(); // 通知待召队列里面的所有其他线程
}
}
}

// 写者代码
… write() {

for(… ) { // 循环执行

synchronized(signal){
while( writers > 0 || readers > 0)
signal.wait();// 如果有人在写或读,那么就放弃执行,进入待召队列

// 能够到达这里,说明没有人在写,也没有人在读

writers ++ ; // 增加一个写者计数,表示本线程在写

// 进行一些读取操作

writers --; // 读取完成,减少一个读者计数,表示本线程不在写

signal.notifyAll(); // 通知待召队列里面的所有其他线程
}
}
}

上述代码只是一段示意代码。实际应用中,人们通常抽取出来一个专门的读写同步锁。
interface ReadWriteLock {
… getReadLock();
… releaseReadLock();
… getWriteLock();
… releaseWriteLock();
}

具体的实现原理也是类似的信号量同步机制。
class RWLock {
… readers, writers;

… synchronized … getReadLock() { // 相当于synchronized(this)

while( writers > 0 )
this.wait(); // 这里我们把RWLock对象本身作为信号量
readers++;
}

…synchronized … releaseReadLock(){ //相当于synchronized(this)
readers--;
this.notifyAll(); // // 这里我们把RWLock对象本身作为信号量
}

…synchronized … getWriteLock(){// 相当于synchronized(this)
while( writers > 0 || readers > 0 )
this.wait(); // 这里我们把RWLock对象本身作为信号量

writers++;
}
…synchronized … releaseWriteLock(){// 相当于synchronized(this)
writers--;
this.notifyAll(); // // 这里我们把RWLock对象本身作为信号量
}
}

具体用法是
public static final RWLock lock = new RWLock();

… read() {
lock.getReadLock();
// 读取
lock.releaseReadLock();
}

… write() {
lock.getWriteLock();
// 读取
lock.releaseWriteLock();
}

这种用法要求在执行一些处理之前,一定要执行某项特殊操作,处理之后一定也要执行某项特殊操作。这种人为的顺序性,无疑增加了代码的耦合度,降低了代码的独立性。很有可能会成为线程死锁和资源操作冲突的根源。
这点一直让我不安,可是没有找到方法避免。毕竟,死锁或者资源操作冲突,是线程的固有问题。
很巧的是,正在我惴惴不安的时候,我的一个朋友提供了一个信息。Sun公司根据JCR,决定在jdk1.5中引入关于concurrency(并发)的部分。
以下这个网址是concurrency部分的util.concurrent一个实现。非常好的信息。对于处理多线程并发问题,很有帮助。
http://gee.cs.oswego.edu/dl/classes/EDU/oswego/cs/dl/util/concurrent/intro.html
里面提供了一个ReadWriteLock类,标准用法如下。
Standard usage of ReadWriteLock:
class X {
ReadWriteLock rw;
// ...
public void read() throws InterruptedException {
rw.readLock().acquire();
try {
// ... do the read
}
finally {
rw.readlock().release()
}
}
public void write() throws InterruptedException {
rw.writeLock().acquire();
try {
// ... do the write
}
finally {
rw.writelock().release()
}
}
}
我们可以看到,ReadWriteLock同样要求调用的顺序——aquire()和release()。我对自己的例子增强了一点信心。
我又查看了WriterPreferenceReadWriteLock类,看到里面成对的方法,startRead(),endRead();startWrite(),endWrite()。我的心情完全放松了下来。我的思路虽然粗糙,但大体的方向是正确的。
分享到:
评论

相关推荐

    多线程不同步读写共享资源代码

    多线程不同步读写共享资源 文章配套代码 我在很早的时候就听说多线程不同步是可以读写共享资源的。这听起来感觉挺好,因为一旦同步线程,将在同步线程上花去一定的CPU时间片. 这一切都是真的,但是,不同步线程的...

    C#多线程读写sqlite

    标题"**C#多线程读写sqlite**"涉及的主要知识点包括: 1. **多线程编程**:C#中的`System.Threading`命名空间提供了丰富的类和方法来创建和管理线程,如`Thread`类、`Task`类以及`ThreadPool`。通过多线程,程序...

    python多线程同步之文件读写控制

    然后,在`file_lock.py`的2.1部分,展示了没有使用任何同步机制(即不加锁)的多线程读写文件。这会导致并发问题,如输出所示,数据被错误地读取和写入,因为多个线程可能同时访问文件,导致数据混乱。 为了修复这...

    秒杀多线程第十六篇 多线程十大经典案例之一 双线程读写队列数据

    在《秒杀多线程系列》的前十五篇中介绍多线程的相关概念,多线程同步互斥问题《秒杀多线程第四篇一个经典的多线程同步问题》及解决多线程同步互斥的常用方法——关键段、事件、互斥量、信号量、读写锁。为了让大家...

    一个多线程同步读写的小程序

    读写同步的实现方式有很多种。一种常见的方法是使用“生产者-消费者”模型,其中写线程作为生产者,负责生成数据(写入),而读线程作为消费者,负责处理这些数据(读取)。在这种模型中,可以使用队列作为缓冲区,...

    delphi多线程 读写数据

    以下是使用TThread进行多线程读写的步骤: 1. **创建TThread子类**:首先,创建一个新的类,继承自TThread。在这个新类中,定义你需要在线程中执行的代码,即读取或写入数据的逻辑。 ```delphi type ...

    多线程通信读写文件

    特别是在处理大量数据时,如读写文件,多线程可以同时执行不同的任务,从而显著提高整体性能。本项目着重探讨如何在多线程环境中实现安全有效的文件读写操作。 首先,我们需要理解多线程的基本概念。在单核处理器...

    用多线程实现串口读写数据以及文件的读写

    为了在多线程环境下安全地读写文件,我们需要考虑线程同步问题。可以通过使用CSingleLock或CCriticalSection等同步对象来确保在任何时候只有一个线程能访问文件,防止数据冲突。 此外,调试时,可以借助串口调试...

    delphi 多线程读写测试

    以下是对"Delphi 多线程读写测试"这个主题的详细解析。 1. **多线程基础**: - **线程定义**:线程是程序执行的最小单元,每个线程都有自己的栈空间,共享同一块内存区域。在单线程程序中,所有任务都在一个线程上...

    多线程读写问题循环buffer

    本示例中的“多线程读写问题循环buffer”着重探讨了如何在多个线程之间安全地共享一个循环缓冲区,以实现高效的数据交换。 首先,我们需要理解“循环buffer”的概念。循环缓冲区是一种内存管理策略,它利用数组或...

    基于多线程和gdal类库的影像读写

    然而,多线程编程也需要注意线程同步问题,防止数据竞争和死锁,确保数据的一致性和完整性。 在用户交互体验方面,本程序还引入了进度条功能。进度条能直观地展示任务的完成状态,给予用户反馈,提高用户体验。在...

    linux下多线程读写socket的简单例子

    在多线程读写Socket的应用中,我们可能会创建一个主线程负责监听和接收连接,然后为每个新的客户端连接创建一个单独的线程来处理读写操作,这样可以避免单线程模型下的阻塞问题,提高服务端的并发处理能力。...

    linux上实现多进程和多线程实现同步互斥(源代码)

    在Linux操作系统中,多进程和多线程是两种并发执行的方式,它们在处理并发问题时,经常需要进行同步和互斥操作,以确保数据的一致性和程序的正确性。本篇将详细介绍这两种并发模型以及如何在Linux环境中实现同步互斥...

    多线程不同步演示2

    本示例"多线程不同步演示2"就是针对这些挑战的一个实例,它通过一个线程写数据,另一个线程读取数据来展示不加同步控制时可能出现的问题。 首先,我们需要理解多线程中的“同步”概念。同步是为了确保多个线程在...

    多线程文件读写测试

    在IT领域,多线程文件读写测试是一个重要的性能优化技术,尤其对于处理大量数据的程序来说至关重要。本文将深入探讨这个主题,并基于提供的文件名列表解析可能的项目结构。 标题"多线程文件读写测试"暗示了这是一个...

    创建多线程线程同步

    4. 文件操作:多线程读写同一文件,需要同步防止数据交错。 总结,多线程和线程同步是提高程序效率和解决并发问题的关键技术。理解并掌握CEVENT对象等同步机制,对于编写高效、可靠的多线程程序至关重要。在实际...

    多线程对文件读写操作(java)

    在Java编程中,多线程技术是提升程序执行效率的关键之一,特别是在进行文件读写操作时。本主题将深入探讨如何使用多线程来优化文件的读取和写入过程,以及需要注意的相关问题。 首先,我们需要理解单线程与多线程在...

    多线程 线程同步

    在计算机科学领域,多线程和线程同步是操作系统中两个关键的概念,它们对于提高程序的执行效率和并发性起着至关重要的作用。在本文中,我们将深入探讨这两个主题,了解其基本概念、应用场景以及相关的内核对象和...

    Python 多线程读写 OPC DA

    本话题聚焦于使用Python进行OPC DA(OPC Data Access)的多线程读写操作。OPC DA是一种标准接口,允许应用程序与工业控制系统设备,如PLC(可编程逻辑控制器)进行通信。这里我们将详细探讨如何利用Python实现这一...

    C#读写线程同步Demo

    4. **ReaderWriterLock和ReaderWriterLockSlim**:这两个类专为读写同步设计,提供读优先的线程同步。允许多个读线程同时访问,但在有写线程时,所有读写线程都会被阻塞。 5. **lock关键字**:这是C#中的一个语法糖...

Global site tag (gtag.js) - Google Analytics