Programowanie D - współbieżność
Współbieżność powoduje, że program działa jednocześnie w wielu wątkach. Przykładem programu współbieżnego jest serwer WWW odpowiadający wielu klientom w tym samym czasie. Współbieżność jest łatwa przy przekazywaniu wiadomości, ale bardzo trudna do napisania, jeśli opiera się na udostępnianiu danych.
Dane przekazywane między wątkami nazywane są wiadomościami. Wiadomości mogą składać się z dowolnego typu i dowolnej liczby zmiennych. Każdy wątek ma identyfikator, który służy do określania odbiorców wiadomości. Każdy wątek, który rozpoczyna inny wątek, jest nazywany właścicielem nowego wątku.
Inicjowanie wątków w D
Funkcja spawn () przyjmuje wskaźnik jako parametr i rozpoczyna nowy wątek z tej funkcji. Wszelkie operacje wykonywane przez tę funkcję, w tym inne funkcje, które może wywołać, zostaną wykonane w nowym wątku. Właściciel i pracownik rozpoczynają wykonywanie oddzielnie, tak jakby byli niezależnymi programami.
Przykład
import std.stdio;
import std.stdio;
import std.concurrency;
import core.thread;
void worker(int a) {
foreach (i; 0 .. 4) {
Thread.sleep(1);
writeln("Worker Thread ",a + i);
}
}
void main() {
foreach (i; 1 .. 4) {
Thread.sleep(2);
writeln("Main Thread ",i);
spawn(≈worker, i * 5);
}
writeln("main is done.");
}
Kiedy powyższy kod jest kompilowany i wykonywany, odczytuje plik utworzony w poprzedniej sekcji i generuje następujący wynik -
Main Thread 1
Worker Thread 5
Main Thread 2
Worker Thread 6
Worker Thread 10
Main Thread 3
main is done.
Worker Thread 7
Worker Thread 11
Worker Thread 15
Worker Thread 8
Worker Thread 12
Worker Thread 16
Worker Thread 13
Worker Thread 17
Worker Thread 18
Identyfikatory wątków w D
ThisTid zmienna ogólnie dostępne na poziomie modułu jest zawsze ID bieżącego wątku. Możesz również otrzymać threadId, gdy zostanie wywołana spawn. Przykład jest pokazany poniżej.
Przykład
import std.stdio;
import std.concurrency;
void printTid(string tag) {
writefln("%s: %s, address: %s", tag, thisTid, &thisTid);
}
void worker() {
printTid("Worker");
}
void main() {
Tid myWorker = spawn(&worker);
printTid("Owner ");
writeln(myWorker);
}
Kiedy powyższy kod jest kompilowany i wykonywany, odczytuje plik utworzony w poprzedniej sekcji i generuje następujący wynik -
Owner : Tid(std.concurrency.MessageBox), address: 10C71A59C
Worker: Tid(std.concurrency.MessageBox), address: 10C71A59C
Tid(std.concurrency.MessageBox)
Przekazywanie wiadomości w D
Funkcja send () wysyła wiadomości, a funkcja otrzymaszOnly () czeka na wiadomość określonego typu. Istnieją inne funkcje o nazwach PrioritySend (), Receive () i ReceiveTimeout (), które zostaną wyjaśnione później.
Właściciel w poniższym programie wysyła swojemu pracownikowi wiadomość typu int i czeka na wiadomość od pracownika typu double. Wątki kontynuują wysyłanie wiadomości tam iz powrotem, dopóki właściciel nie wyśle negatywnej int. Przykład jest pokazany poniżej.
Przykład
import std.stdio;
import std.concurrency;
import core.thread;
import std.conv;
void workerFunc(Tid tid) {
int value = 0;
while (value >= 0) {
value = receiveOnly!int();
auto result = to!double(value) * 5; tid.send(result);
}
}
void main() {
Tid worker = spawn(&workerFunc,thisTid);
foreach (value; 5 .. 10) {
worker.send(value);
auto result = receiveOnly!double();
writefln("sent: %s, received: %s", value, result);
}
worker.send(-1);
}
Kiedy powyższy kod jest kompilowany i wykonywany, odczytuje plik utworzony w poprzedniej sekcji i generuje następujący wynik -
sent: 5, received: 25
sent: 6, received: 30
sent: 7, received: 35
sent: 8, received: 40
sent: 9, received: 45
Przekazywanie wiadomości z czekaniem w D
Poniżej przedstawiono prosty przykład z komunikatem przekazywanym z czekaniem.
import std.stdio;
import std.concurrency;
import core.thread;
import std.conv;
void workerFunc(Tid tid) {
Thread.sleep(dur!("msecs")( 500 ),);
tid.send("hello");
}
void main() {
spawn(&workerFunc,thisTid);
writeln("Waiting for a message");
bool received = false;
while (!received) {
received = receiveTimeout(dur!("msecs")( 100 ), (string message) {
writeln("received: ", message);
});
if (!received) {
writeln("... no message yet");
}
}
}
Kiedy powyższy kod jest kompilowany i wykonywany, odczytuje plik utworzony w poprzedniej sekcji i generuje następujący wynik -
Waiting for a message
... no message yet
... no message yet
... no message yet
... no message yet
received: hello