Pemrograman D - Konkurensi

Concurrency adalah membuat program berjalan di banyak utas sekaligus. Contoh program konkuren adalah server web yang merespons banyak klien pada waktu yang sama. Konkurensi mudah dengan penyampaian pesan tetapi sangat sulit untuk ditulis jika didasarkan pada berbagi data.

Data yang dilewatkan di antara utas disebut pesan. Pesan dapat terdiri dari semua jenis dan sejumlah variabel. Setiap utas memiliki id, yang digunakan untuk menentukan penerima pesan. Setiap utas yang memulai utas lain disebut pemilik utas baru.

Memulai Thread di D

Fungsi spawn () mengambil pointer sebagai parameter dan memulai utas baru dari fungsi itu. Setiap operasi yang dilakukan oleh fungsi itu, termasuk fungsi lain yang mungkin dipanggil, akan dijalankan pada thread baru. Pemilik dan pekerja sama-sama mulai mengeksekusi secara terpisah seolah-olah mereka adalah program independen.

Contoh

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.");  
}

Ketika kode di atas dikompilasi dan dijalankan, ia membaca file yang dibuat di bagian sebelumnya dan menghasilkan hasil sebagai berikut -

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

Pengidentifikasi Benang di D

The thisTid variabel yang tersedia secara global pada tingkat modul selalu id dari thread saat ini. Anda juga dapat menerima threadId saat pemijahan dipanggil. Contohnya ditunjukkan di bawah ini.

Contoh

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); 
}

Ketika kode di atas dikompilasi dan dijalankan, ia membaca file yang dibuat di bagian sebelumnya dan menghasilkan hasil sebagai berikut -

Owner : Tid(std.concurrency.MessageBox), address: 10C71A59C 
Worker: Tid(std.concurrency.MessageBox), address: 10C71A59C 
Tid(std.concurrency.MessageBox)

Message Passing in D

Fungsi send () mengirim pesan dan fungsi acceptOnly () menunggu pesan dari jenis tertentu. Ada fungsi lain bernama prioritySend (), accept (), dan acceptTimeout (), yang akan dijelaskan nanti.

Pemilik dalam program berikut mengirimkan pekerjanya pesan tipe int dan menunggu pesan dari pekerja berjenis double. Utas terus mengirim pesan bolak-balik sampai pemilik mengirim int negatif. Contohnya ditunjukkan di bawah ini.

Contoh

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); 
}

Ketika kode di atas dikompilasi dan dijalankan, ia membaca file yang dibuat di bagian sebelumnya dan menghasilkan hasil sebagai berikut -

sent: 5, received: 25 
sent: 6, received: 30 
sent: 7, received: 35 
sent: 8, received: 40 
sent: 9, received: 45

Message Passing dengan Wait in D

Contoh sederhana dengan pesan yang lewat dengan menunggu ditampilkan di bawah ini.

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"); 
      }
   } 
}

Ketika kode di atas dikompilasi dan dijalankan, ia membaca file yang dibuat di bagian sebelumnya dan menghasilkan hasil sebagai berikut -

Waiting for a message 
... no message yet 
... no message yet 
... no message yet 
... no message yet 
received: hello