RwLock

Struct RwLock 

Source
pub struct RwLock<T> { /* private fields */ }
Expand description

A Read-Write Lock.

This lock is similar to the std::sync::RwLock. But it is distributed over multiple instances of the same service.

§Threads

If you try to get a writer lock in a thread, which already has a reader lock, you will end up in a deadlock. To use the RwLock in threads, you need a scoped thread.

§Examples

§Linear usage

use dtypes::redis::sync::RwLock;
use dtypes::redis::types::Di32;
use std::thread;

let client = redis::Client::open("redis://localhost:6379").unwrap();
let client2 = client.clone();
let mut i32 = Di32::with_value(1, "test_rwlock_example1", client.clone());
let mut lock = RwLock::new(i32);

// many reader locks can be held at once
{
    let read1 = lock.read().unwrap();
    let read2 = lock.read().unwrap();
    assert_eq!(*read1, 1);
} // read locks are dropped at this point

// only one writer lock can be held, however
{
    let mut write1 = lock.write().unwrap();
    write1.store(2).unwrap();
    assert_eq!(*write1, 2);
    // the next line is not allowed, because it deadlocks.
    //let mut _ = lock.write().unwrap();
} // write lock is dropped here

// look, you can read it again
{
   let read1 = lock.read().unwrap();
   assert_eq!(*read1, 2);
}

§Threaded usage

use dtypes::redis::sync::RwLock;
use dtypes::redis::types::Di32;
use std::thread;

let client = redis::Client::open("redis://localhost:6379").unwrap();
let i32 = Di32::with_value(1, "test_rwlock_example2", client.clone());
let mut lock = RwLock::new(i32);
// the reader lock is dropped immediately
assert_eq!(*lock.read().unwrap(), 1);
// Scoped threads are needed, otherwise the lifetime is unclear.
thread::scope(|s| {
       s.spawn(|| {
           let mut write = lock.write().unwrap();
           write.store(2);
           assert_eq!(*write, 2);
       }).join().unwrap();
});
assert_eq!(*lock.read().unwrap(), 2);

Implementations§

Source§

impl<T> RwLock<T>

Source

pub fn new(data: Generic<T>) -> Self

Source

pub fn read(&self) -> Result<RwLockReadGuard<'_, T>, LockError>

Creates a new RwLock Reader.

This function blocks until the lock is acquired. If there is a writer lock, this function blocks until the writer lock is dropped. Also if there is a writer locks waiting to be acquired, this function blocks until the writer lock is acquired and dropped.

Source

pub fn write(&mut self) -> Result<RwLockWriteGuard<'_, T>, LockError>

Creates a new RwLock Writer.

This function blocks until the lock is acquired. If there is a reader lock, this function blocks until the reader lock is dropped. The acquiring writer lock has priority over any waiting reader lock.

Methods from Deref<Target = Generic<T>>§

Source

pub fn store(&mut self, value: T)

The store method sets the value of the type.

Examples found in repository?
examples/2services.rs (line 15)
5fn main() {
6    thread::scope(|s| {
7        let client = redis::Client::open("redis://localhost:6379").unwrap();
8        let client2 = client.clone();
9
10        let t1 = s.spawn(move || {
11            let mut string = String::with_value("Hello".to_string(), "test", client);
12            println!("Thread1: {}", string.cached().unwrap());
13            assert_eq!(string, "Hello");
14            sleep(std::time::Duration::from_secs(1));
15            string.store("World".to_string());
16            println!("Thread1: {}", string.cached().unwrap());
17            assert_eq!(string, "World");
18        });
19
20        let t2 = s.spawn(move || {
21            sleep(std::time::Duration::from_micros(100));
22            let mut string = String::with_load("test", client2);
23            println!("Thread2: {}", string.cached().unwrap());
24            assert_eq!(string, "Hello");
25            sleep(std::time::Duration::from_secs(2));
26            string.acquire();
27            println!("Thread2: {}", string.cached().unwrap());
28            assert_eq!(string, "World");
29        });
30        t1.join().expect("Failed to join thread1");
31        t2.join().expect("Failed to join thread2");
32    });
33}
Source

pub fn acquire(&mut self) -> &T

The acquire method returns a reference to the value stored in the type. Loads it from the redis directly.

§Example
use dtypes::redis::types::Di32 as i32;

let client = redis::Client::open("redis://localhost:6379").unwrap();
let mut i32 = i32::with_value(1, "test_add", client.clone());
i32 = i32 + i32::with_value(2, "test_add2", client);
assert_eq!(i32.acquire(), &3);
Examples found in repository?
examples/2services.rs (line 26)
5fn main() {
6    thread::scope(|s| {
7        let client = redis::Client::open("redis://localhost:6379").unwrap();
8        let client2 = client.clone();
9
10        let t1 = s.spawn(move || {
11            let mut string = String::with_value("Hello".to_string(), "test", client);
12            println!("Thread1: {}", string.cached().unwrap());
13            assert_eq!(string, "Hello");
14            sleep(std::time::Duration::from_secs(1));
15            string.store("World".to_string());
16            println!("Thread1: {}", string.cached().unwrap());
17            assert_eq!(string, "World");
18        });
19
20        let t2 = s.spawn(move || {
21            sleep(std::time::Duration::from_micros(100));
22            let mut string = String::with_load("test", client2);
23            println!("Thread2: {}", string.cached().unwrap());
24            assert_eq!(string, "Hello");
25            sleep(std::time::Duration::from_secs(2));
26            string.acquire();
27            println!("Thread2: {}", string.cached().unwrap());
28            assert_eq!(string, "World");
29        });
30        t1.join().expect("Failed to join thread1");
31        t2.join().expect("Failed to join thread2");
32    });
33}
Source

pub fn cached(&self) -> Option<&T>

The get method returns a reference to the value stored in the type.

Examples found in repository?
examples/2services.rs (line 12)
5fn main() {
6    thread::scope(|s| {
7        let client = redis::Client::open("redis://localhost:6379").unwrap();
8        let client2 = client.clone();
9
10        let t1 = s.spawn(move || {
11            let mut string = String::with_value("Hello".to_string(), "test", client);
12            println!("Thread1: {}", string.cached().unwrap());
13            assert_eq!(string, "Hello");
14            sleep(std::time::Duration::from_secs(1));
15            string.store("World".to_string());
16            println!("Thread1: {}", string.cached().unwrap());
17            assert_eq!(string, "World");
18        });
19
20        let t2 = s.spawn(move || {
21            sleep(std::time::Duration::from_micros(100));
22            let mut string = String::with_load("test", client2);
23            println!("Thread2: {}", string.cached().unwrap());
24            assert_eq!(string, "Hello");
25            sleep(std::time::Duration::from_secs(2));
26            string.acquire();
27            println!("Thread2: {}", string.cached().unwrap());
28            assert_eq!(string, "World");
29        });
30        t1.join().expect("Failed to join thread1");
31        t2.join().expect("Failed to join thread2");
32    });
33}

Trait Implementations§

Source§

impl<T> Deref for RwLock<T>

Source§

type Target = Generic<T>

The resulting type after dereferencing.
Source§

fn deref(&self) -> &Self::Target

Dereferences the value.
Source§

impl<T> DerefMut for RwLock<T>

Source§

fn deref_mut(&mut self) -> &mut Self::Target

Mutably dereferences the value.

Auto Trait Implementations§

§

impl<T> Freeze for RwLock<T>
where T: Freeze,

§

impl<T> !RefUnwindSafe for RwLock<T>

§

impl<T> Send for RwLock<T>
where T: Send,

§

impl<T> Sync for RwLock<T>
where T: Sync,

§

impl<T> Unpin for RwLock<T>
where T: Unpin,

§

impl<T> !UnwindSafe for RwLock<T>

Blanket Implementations§

Source§

impl<T> Any for T
where T: 'static + ?Sized,

Source§

fn type_id(&self) -> TypeId

Gets the TypeId of self. Read more
Source§

impl<T> Borrow<T> for T
where T: ?Sized,

Source§

fn borrow(&self) -> &T

Immutably borrows from an owned value. Read more
Source§

impl<T> BorrowMut<T> for T
where T: ?Sized,

Source§

fn borrow_mut(&mut self) -> &mut T

Mutably borrows from an owned value. Read more
Source§

impl<T> From<T> for T

Source§

fn from(t: T) -> T

Returns the argument unchanged.

Source§

impl<T, U> Into<U> for T
where U: From<T>,

Source§

fn into(self) -> U

Calls U::from(self).

That is, this conversion is whatever the implementation of From<T> for U chooses to do.

Source§

impl<P, T> Receiver for P
where P: Deref<Target = T> + ?Sized, T: ?Sized,

Source§

type Target = T

🔬This is a nightly-only experimental API. (arbitrary_self_types)
The target type on which the method may be called.
Source§

impl<T, U> TryFrom<U> for T
where U: Into<T>,

Source§

type Error = Infallible

The type returned in the event of a conversion error.
Source§

fn try_from(value: U) -> Result<T, <T as TryFrom<U>>::Error>

Performs the conversion.
Source§

impl<T, U> TryInto<U> for T
where U: TryFrom<T>,

Source§

type Error = <U as TryFrom<T>>::Error

The type returned in the event of a conversion error.
Source§

fn try_into(self) -> Result<U, <U as TryFrom<T>>::Error>

Performs the conversion.