Chinaunix首页 | 论坛 | 博客
  • 博客访问: 45817
  • 博文数量: 29
  • 博客积分: 2000
  • 博客等级: 大尉
  • 技术积分: 330
  • 用 户 组: 普通用户
  • 注册时间: 2007-09-20 14:50
文章分类

全部博文(29)

文章存档

2011年(1)

2008年(4)

2007年(24)

我的朋友
最近访客

分类: WINDOWS

2007-09-27 14:52:36

Using Event Objects
Applications use event objects in a number of situations to notify a waiting thread of the occurrence of an event. For example, overlapped I/O operations on files, named pipes, and communications devices use an event object to signal their completion. For more information about the use of event objects in overlapped I/O operations, see Synchronization and Overlapped Input and Output.
The following example uses event objects to prevent several threads from reading from a shared memory buffer while a master thread is writing to that buffer. First, the master thread uses the CreateEvent function to create a manual-reset event object. The master thread sets the event object to nonsignaled when it is writing to the buffer and then resets the object to signaled when it has finished writing. Then it creates several reader threads and an auto-reset event object for each thread. Each reader thread sets its event object to signaled when it is not reading from the buffer.
Before the master thread writes to the shared buffer, it uses the ResetEvent function to set the state of ghGlobalWriteEvent (an application-defined global variable) to nonsignaled. This blocks the reader threads from starting a read operation. The master then uses the WaitForMultipleObjects function to wait for all reader threads to finish any current read operations. When WaitForMultipleObjects returns, the master thread can safely write to the buffer. After it has finished, it sets ghGlobalWriteEvent and all the reader-thread events to signaled, enabling the reader threads to start new read operations.
Before starting a read operation, each reader thread uses WaitForMultipleObjects to wait for ghGlobalWriteEvent and its own read event to be signaled. When WaitForMultipleObjects returns, the reader thread's auto-reset event has been reset to nonsignaled. This blocks the master thread from writing to the buffer until the reader thread uses the SetEvent function to set the event's state back to signaled.

 Copy Code
#include
#include
#define THREADCOUNT 4
HANDLE ghGlobalWriteEvent;
HANDLE ghReadEvents[THREADCOUNT];
DWORD WINAPI ThreadProc(LPVOID);
void CreateEventsAndThreads(void)
{
    HANDLE hThread;
    DWORD i, dwThreadID;
    // Create a manual-reset event object. The master thread sets
    // this to nonsignaled when it writes to the shared buffer.
    ghGlobalWriteEvent = CreateEvent(
        NULL,               // default security attributes
        TRUE,               // manual-reset event
        TRUE,               // initial state is signaled
        TEXT("WriteEvent")  // object name
        );
    if (ghGlobalWriteEvent == NULL)
    {
        printf("CreateEvent failed (%d)\n", GetLastError());
        return;
    }
    else if ( GetLastError() == ERROR_ALREADY_EXISTS )
    {
        printf("Named event already exists.\n");
        return;
    }
    // Create multiple threads and an auto-reset event object
    // for each thread. Each thread sets its event object to
    // signaled when it is not reading from the shared buffer.
    for(i = 0; i < THREADCOUNT; i++)
    {
        // Create the auto-reset event
        ghReadEvents[i] = CreateEvent(
            NULL,     // no security attributes
            FALSE,    // auto-reset event
            TRUE,     // initial state is signaled
            NULL);    // object not named
        if (ghReadEvents[i] == NULL)
        {
            printf("CreateEvent failed (%d)\n", GetLastError());
            return;
        }
        hThread = CreateThread(NULL,
            0,
            ThreadProc,
            &ghReadEvents[i],  // pass event handle
            0,
            &dwThreadID);
        if (hThread == NULL)
        {
            printf("CreateThread failed (%d)\n", GetLastError());
            return;
        }
    }
}
void WriteToBuffer(VOID)
{
    DWORD dwWaitResult, i;
    // Reset ghGlobalWriteEvent to nonsignaled, to block readers
 
    if (! ResetEvent(ghGlobalWriteEvent) )
    {
        printf("ResetEvent failed (%d)\n", GetLastError());
        return;
    }
    // Wait for all reading threads to finish reading
    dwWaitResult = WaitForMultipleObjects(
        THREADCOUNT,   // number of handles in array
        ghReadEvents,  // array of read-event handles
        TRUE,          // wait until all are signaled
        INFINITE);     // indefinite wait
    switch (dwWaitResult)
    {
        // All read-event objects were signaled
        case WAIT_OBJECT_0:
            // TODO: Write to the shared buffer
            printf("Main thread writing to the shared buffer...\n");
            break;
        // An error occurred
        default:
            printf("Wait error: %d\n", GetLastError());
            ExitProcess(0);
    }
    // Set ghGlobalWriteEvent to signaled
    if (! SetEvent(ghGlobalWriteEvent) )
    {
        printf("SetEvent failed (%d)\n", GetLastError());
        ExitProcess(0);
    }
    // Set all read events to signaled
    for(i = 0; i < THREADCOUNT; i++)
        if (! SetEvent(ghReadEvents[i]) )
        {
            printf("SetEvent failed (%d)\n", GetLastError());
            return;
        }
}
void CloseEvents()
{
    int i;
    for( i=0; i < THREADCOUNT; i++ )
        CloseHandle(ghReadEvents[i]);
    CloseHandle(ghGlobalWriteEvent);
}
void main()
{
    int i;
    // TODO: Create the shared buffer
    // Create the events and THREADCOUNT threads to read from the buffer
    CreateEventsAndThreads();
    // Write to the buffer three times, just for test purposes
    for(i=0; i < 3; i++)
        WriteToBuffer();
    // Close the events
    CloseEvents();
}
DWORD WINAPI ThreadProc(LPVOID lpParam)
{
    DWORD dwWaitResult;
    HANDLE hEvents[2];
    hEvents[0] = *(HANDLE*)lpParam;  // thread's read event
    hEvents[1] = ghGlobalWriteEvent; // global write event
    dwWaitResult = WaitForMultipleObjects(
        2,            // number of handles in array
        hEvents,      // array of event handles
        TRUE,         // wait till all are signaled
        INFINITE);    // indefinite wait
    switch (dwWaitResult)
    {
        // Both event objects were signaled
        case WAIT_OBJECT_0:
            // TODO: Read from the shared buffer
            printf("Thread %d reading from buffer...\n",
                   GetCurrentThreadId());
            break;
        // An error occurred
        default:
            printf("Wait error: %d\n", GetLastError());
            ExitThread(0);
    }
    // Set the read event to signaled
    if (! SetEvent(hEvents[0]) )
    {
        printf("SetEvent failed (%d)\n", GetLastError());
        ExitThread(0);
    }
    return 1;
}
阅读(259) | 评论(0) | 转发(0) |
0

上一篇:第一篇blog

下一篇:20070927

给主人留下些什么吧!~~