rust/src/libcore/future.rs

176 lines
3.3 KiB
Rust
Raw Normal View History

#[doc = "
A type representing values that may be computed concurrently and
operations for working with them.
Example:
> let delayed_fib = future::spawn {|| fib(5000) };
> make_a_sandwitch();
> io::println(#fmt(\"fib(5000) = %?\", delayed_fib.get()))
"];
export future;
export future::{};
export from_value;
export from_port;
export from_fn;
export get;
export with;
export spawn;
import either = either::t;
#[doc = "The future type"]
enum future<A> = {
mutable v: either<@A, fn@() -> A>
};
#[doc = "Methods on the `future` type"]
impl future<A:send> for future<A> {
fn get() -> A {
#[doc = "Get the value of the future"];
get(self)
}
fn with<B>(blk: fn(A) -> B) -> B {
#[doc = "Work with the value without copying it"];
with(self, blk)
}
}
fn from_value<A>(+val: A) -> future<A> {
#[doc = "
Create a future from a value. The value is immediately available
and calling `get` later will not block.
"];
future({
mutable v: either::left(@val)
})
}
fn from_port<A:send>(-port: comm::port<A>) -> future<A> {
#[doc = "
Create a future from a port. The first time that the value is
requested the task will block waiting for the result to be
received on the port.
"];
from_fn {||
comm::recv(port)
}
}
fn from_fn<A>(f: fn@() -> A) -> future<A> {
#[doc = "
Create a future from a function. The first time that the value is
requested it will be retreived by calling the function.
Note that this function is a local function. It is not spawned into
another task.
"];
future({
mutable v: either::right(f)
})
}
fn spawn<A:send>(+blk: fn~() -> A) -> future<A> {
#[doc = "
Create a future from a unique closure. The closure will be run
in a new task and its result used as the value of the future.
"];
let po = comm::port();
let ch = comm::chan(po);
task::spawn {||
comm::send(ch, blk())
};
from_port(po)
}
fn get<A:copy>(future: future<A>) -> A {
#[doc = "Get the value of the future"];
with(future) {|v| v }
}
fn with<A,B>(future: future<A>, blk: fn(A) -> B) -> B {
#[doc = "Work with the value without copying it"];
let v = alt future.v {
either::left(v) { v }
either::right(f) {
let v = @f();
future.v = either::left(v);
v
}
};
blk(*v)
}
#[test]
fn test_from_value() {
let f = from_value("snail");
assert get(f) == "snail";
}
#[test]
fn test_from_port() {
let po = comm::port();
let ch = comm::chan(po);
comm::send(ch, "whale");
let f = from_port(po);
assert get(f) == "whale";
}
#[test]
fn test_from_fn() {
let f = fn@() -> str { "brail" };
let f = from_fn(f);
assert get(f) == "brail";
}
#[test]
fn test_iface_get() {
let f = from_value("fail");
assert f.get() == "fail";
}
#[test]
fn test_with() {
let f = from_value("nail");
assert with(f) {|v| v} == "nail";
}
#[test]
fn test_iface_with() {
let f = from_value("kale");
assert f.with {|v| v} == "kale";
}
#[test]
fn test_spawn() {
let f = spawn {|| "bale" };
assert get(f) == "bale";
}
#[test]
#[should_fail]
#[ignore(cfg(target_os = "win32"))]
fn test_futurefail() {
let f = spawn {|| fail };
let _x: str = get(f);
}