D 프로그래밍-동시성

동시성은 프로그램이 한 번에 여러 스레드에서 실행되도록합니다. 동시 프로그램의 예로는 동시에 많은 클라이언트에 응답하는 웹 서버가 있습니다. 동시성은 메시지 전달과 함께 쉽지만 데이터 공유를 기반으로하는 경우 쓰기가 매우 어렵습니다.

스레드간에 전달되는 데이터를 메시지라고합니다. 메시지는 모든 유형과 여러 변수로 구성 될 수 있습니다. 모든 스레드에는 메시지 수신자를 지정하는 데 사용되는 ID가 있습니다. 다른 스레드를 시작하는 모든 스레드를 새 스레드의 소유자라고합니다.

D에서 스레드 시작

spawn () 함수는 포인터를 매개 변수로 사용하고 해당 함수에서 새 스레드를 시작합니다. 호출 할 수있는 다른 함수를 포함하여 해당 함수에 의해 수행되는 모든 작업은 새 스레드에서 실행됩니다. 소유자와 작업자는 마치 독립적 인 프로그램 인 것처럼 개별적으로 실행을 시작합니다.

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

위의 코드가 컴파일되고 실행되면 이전 섹션에서 생성 된 파일을 읽고 다음과 같은 결과를 생성합니다.

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

D의 스레드 식별자

thisTid의 모듈 수준에서 변수를 사용할 세계적으로는 항상 현재의 thread의 ID입니다. 또한 spawn이 호출 될 때 threadId를받을 수 있습니다. 아래에 예가 나와 있습니다.

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

위의 코드가 컴파일되고 실행되면 이전 섹션에서 생성 된 파일을 읽고 다음과 같은 결과를 생성합니다.

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

D로 전달되는 메시지

send () 함수는 메시지를 보내고 receiveOnly () 함수는 특정 유형의 메시지를 기다립니다. prioritySend (), receive () 및 receiveTimeout ()이라는 다른 함수가 있습니다. 이에 대해서는 나중에 설명합니다.

다음 프로그램의 소유자는 작업자에게 int 유형의 메시지를 보내고 double 유형의 작업자로부터 메시지를 기다립니다. 스레드는 소유자가 음의 정수를 보낼 때까지 메시지를 계속해서 앞뒤로 보냅니다. 아래에 예가 나와 있습니다.

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

위의 코드가 컴파일되고 실행되면 이전 섹션에서 생성 된 파일을 읽고 다음과 같은 결과를 생성합니다.

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

D에서 Wait로 전달되는 메시지

wait 메시지가 전달되는 간단한 예가 아래에 나와 있습니다.

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

위의 코드가 컴파일되고 실행되면 이전 섹션에서 생성 된 파일을 읽고 다음과 같은 결과를 생성합니다.

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