Condividi tramite


Classe CEvent

Rappresenta un evento, un oggetto di sincronizzazione che consente a un thread per notificare a un altro che un evento.

class CEvent : public CSyncObject

Membri

Costruttori pubblici

Nome

Descrizione

CEvent::CEvent

Costruisce un oggetto CEvent.

Metodi pubblici

Nome

Descrizione

CEvent::PulseEvent

Imposta l'evento su disponibile (segnalato), le versioni thread in attesa e imposta l'evento su non disponibile (non segnalato).

CEvent::ResetEvent

Imposta l'evento su non disponibile (non segnalato).

CEvent::SetEvent

Imposta l'evento su disponibile (segnalato) e versioni tutti i thread in attesa.

CEvent::Unlock

Rilascia l'oggetto evento.

Note

Gli eventi sono utili quando un thread deve sapere quando eseguire la relativa attività. Ad esempio, un thread che copia dei dati a un archivio dati deve ricevere una notifica quando i nuovi dati disponibili. Utilizzando un oggetto CEvent per notificare alla copia thread quando i nuovi dati sono disponibili, il thread può eseguire la relativa attività il prima possibile.

Gli oggetti diCEvent dispongono di due tipi: manuale e automatico.

Restituisce automatici di un oggetto CEvent automaticamente a uno stato non disponibile) non segnalato almeno un thread viene rilasciato. Per impostazione predefinita, un oggetto CEvent è automatico a meno che non venga passato TRUE per il parametro bManualReset durante la costruzione.

Un oggetto CEvent manuali rimane nello stato impostato da SetEvent o da ResetEvent fino a chiamare un'altra funzione. Per creare un oggetto manuale CEvent, passare TRUE per il parametro bManualReset durante la costruzione.

Per utilizzare un oggetto CEvent, costruire l'oggetto CEvent quando necessario. Specificare il nome dell'evento che si desidera attendere su e specificare che l'applicazione deve inizialmente possederla. È quindi possibile accedere all'evento al completamento del costruttore. Chiamare SetEvent per segnalare (rendere) l'oggetto evento e quindi chiamare Sblocca al termine di accedere alla risorsa controllata.

Un metodo alternativo per gli oggetti CEvent è possibile aggiungere una variabile di tipo CEvent come un membro dati alla classe che si desidera controllare. Durante la costruzione dell'oggetto selezionato, chiamare il costruttore del membro dati CEvent e specificare se l'evento è inizialmente segnalato nonché tipo di specifythe di oggetto evento desiderato, il nome dell'evento (se verrà utilizzato oltre i limiti dei processi) e di tutti gli attributi di sicurezza desiderate.

Per accedere a una risorsa controllata da un oggetto CEvent in questo modo, creare innanzitutto una variabile di tipo CSingleLock o digitare CMultiLock nel metodo di accesso alla risorsa. Chiamare il metodo Lock dell'oggetto di blocco (ad esempio, CMultiLock::Lock). In questa fase, il thread o accederà alla risorsa, attendere che la risorsa da rilasciare e accederà, o attendere che la risorsa da rilasciare, è scaduto e non riesce ad accedere alla risorsa. Tuttavia, la risorsa è stata accedere in modo thread-safe. Per rilasciare la risorsa, chiamare SetEvent per segnalare l'oggetto evento e quindi il metodo Unlock dell'oggetto di blocco (ad esempio, CMultiLock::Unlock, oppure lasciare il blocco nella UIambito.

Per ulteriori informazioni su come utilizzare gli oggetti CEvent, vedere Multithreading: utilizzo delle classi di sincronizzazione.

Esempio

// The following demonstrates trivial usage of the CEvent class. 
// A CEvent object is created and passed as a parameter to another  
// thread.  The other thread will wait for the event to be signaled 
// and then exit

UINT __cdecl MyThreadProc(LPVOID lpParameter)
{
   CEvent* pEvent = (CEvent*)(lpParameter);
   VERIFY(pEvent != NULL);

   // Wait for the event to be signaled
   ::WaitForSingleObject(pEvent->m_hObject, INFINITE);

   // Terminate the thread
   ::AfxEndThread(0, FALSE); 
   return 0L;
}

void CEvent_Test()
{
   // Create the CEvent object that will be passed to the thread routine
   CEvent* pEvent = new CEvent(FALSE, FALSE);

   // Create a thread that will wait on the event
   CWinThread* pThread;
   pThread = ::AfxBeginThread(&MyThreadProc, pEvent, 0, 0, CREATE_SUSPENDED, NULL);
   pThread->m_bAutoDelete = FALSE; 
   pThread->ResumeThread();

   // Signal the thread to do the next work item
   pEvent->SetEvent();

   // Wait for the thread to consume the event and return
   ::WaitForSingleObject(pThread->m_hThread, INFINITE); 
   delete pThread;
   delete pEvent;
}
// This example builds upon the previous one. 
// A second thread is created to calculate prime numbers. 
// The main thread will signal the second thread to calulate the next  
// prime number in the series.  The second thread signals the first  
// after each number is calculated. Finally, after several iterations  
// the worker thread is signaled to terminate. 

class CPrimeTest
{
public:
   CPrimeTest()
      : m_pCalcNext(new CEvent(FALSE, FALSE))
      , m_pCalcFinished(new CEvent(FALSE, FALSE))
      , m_pTerminateThread(new CEvent(FALSE, FALSE))
      , m_iCurrentPrime(0)
   {   
      // Create a thread that will calculate the prime numbers
      CWinThread* pThread;
      pThread = ::AfxBeginThread(&PrimeCalcProc, this, 0, 0, CREATE_SUSPENDED, NULL);
      pThread->m_bAutoDelete = FALSE; 
      pThread->ResumeThread();

      // Calcuate the first 10 prime numbers in the series on the thread 
      for(UINT i = 0; i < 10; i++)
      {
         // Signal the thread to do the next work item
         m_pCalcNext->SetEvent();
         // Wait for the thread to complete the current task
         ::WaitForSingleObject(m_pCalcFinished->m_hObject, INFINITE);
         // Print the result
         TRACE(_T("The value of m_iCurrentPrime is: %d\n"), m_iCurrentPrime);
      }

      // Notify the worker thread to exit and wait for it to complete
      m_pTerminateThread->SetEvent();
      ::WaitForSingleObject(pThread->m_hThread, INFINITE); 
      delete pThread;
   }
   ~CPrimeTest()
   {
      delete m_pCalcNext;
      delete m_pCalcFinished;
      delete m_pTerminateThread;
   }

private:
   // Determines whether the given number is a prime number 
   static BOOL IsPrime(INT ThisPrime)
   {
      if(ThisPrime < 2) 
         return FALSE;

      for(INT n = 2; n < ThisPrime; n++)
      {
         if(ThisPrime % n == 0)
            return FALSE;
      }
      return TRUE;
   }

   // Calculates the next prime number in the series 
   static INT NextPrime(INT ThisPrime)
   {
      while(TRUE)
      {
         if(IsPrime(++ThisPrime))
         {
            return ThisPrime;
         }
      }
   }

   // Worker thread responsible for calculating the next prime 
   // number in the series 
   static UINT __cdecl PrimeCalcProc(LPVOID lpParameter)
   {
      CPrimeTest* pThis = static_cast<CPrimeTest*>(lpParameter);
      VERIFY(pThis != NULL);

      VERIFY(pThis->m_pCalcNext != NULL);
      VERIFY(pThis->m_pCalcFinished != NULL);
      VERIFY(pThis->m_pTerminateThread != NULL);

      // Create a CMultiLock object to wait on the various events 
      // WAIT_OBJECT_0 refers to the first event in the array, WAIT_OBJECT_0+1 refers to the second
      CSyncObject* pWaitObjects[] = { pThis->m_pCalcNext, pThis->m_pTerminateThread };
      CMultiLock MultiLock(pWaitObjects, 2L);
      while(MultiLock.Lock(INFINITE, FALSE) == WAIT_OBJECT_0) 
      {         
         // Calculate next prime
         pThis->m_iCurrentPrime = NextPrime(pThis->m_iCurrentPrime);
         // Notify main thread calculation is complete
         pThis->m_pCalcFinished->SetEvent();
       } 

      // Terminate the thread
       ::AfxEndThread(0, FALSE); 
      return 0L;
   }

   CEvent* m_pCalcNext;      // notifies worker thread to calculate next prime
   CEvent* m_pCalcFinished;   // notifies main thread current calculation is complete
   CEvent* m_pTerminateThread;   // notifies worker thread to terminate

   INT m_iCurrentPrime;   // current calculated prime number
};

Gerarchia di ereditarietà

CObject

CSyncObject

CEvent

Requisiti

Header: afxmt.h

Vedere anche

Riferimenti

Classe CSyncObject

Grafico delle gerarchie