Студопедия  
Главная страница | Контакты | Случайная страница

АвтомобилиАстрономияБиологияГеографияДом и садДругие языкиДругоеИнформатика
ИсторияКультураЛитератураЛогикаМатематикаМедицинаМеталлургияМеханика
ОбразованиеОхрана трудаПедагогикаПолитикаПравоПсихологияРелигияРиторика
СоциологияСпортСтроительствоТехнологияТуризмФизикаФилософияФинансы
ХимияЧерчениеЭкологияЭкономикаЭлектроника

Взаимоисключения

Объекты-взаимоисключения (мьютексы, mutex - от MUTual EXclusion) позволяют координировать взаимное исключение доступа к разделяемому ресурсу. Сигнальное состояние объекта (т.е. состояние "установлен") соответствует моменту времени, когда объект не принадлежит ни одной нити и его можно "захватить". И наоборот, состояние "сброшен" (не сигнальное) соответствует моменту, когда какая-либо нить уже владеет этим объектом. Доступ к объекту разрешается, когда нить, владеющая объектом, освободит его.

Две (или более) нити могут создать мьютекс с одним и тем же именем, вызвав функцию CreateMutex. Первая нить действительно создает мьютекс, а следующие - получают дескриптор уже существующего объекта. Это дает возможность нескольким нитям получить дескриптор одного и того же мьютекса, освобождая программиста от необходимости заботиться о том, кто в действительности создает мьютекс. Если используется такой подход, желательно установить флаг bInitialOwner в FALSE, иначе возникнут определенные трудности при определении действительного создателя мьютекса.

Несколько нитей могут получить дескриптор одного и того же мьютекса, что делает возможным взаимодействие между процессами. Можно использовать следующие механизмы такого подхода:

Для того чтобы объявить взаимоисключение принадлежащим текущей нити, надо вызвать одну из ожидающих функций. Нить, которой принадлежит объект, может его "захватывать" повторно сколько угодно раз (это не приведет к самоблокировке), но столько же раз она должна будет его освобождать с помощью функции ReleaseMutex.

Для синхронизации нитей одного процесса более эффективно использование критических секций.

 

 

Пример. Синхронизация нитей с помощью мьютексов.

#include <windows.h>

#include <stdio.h>

HANDLE hMutex;

int a[5];

HANDLE hThr;

unsigned long uThrID;

void Thread(void* pParams)

{

int i, num = 0;

while (1)

{

WaitForSingleObject(hMutex, INFINITE);

for (i=0; i<5; i++) a[i] = num;

num++;

ReleaseMutex(hMutex);

}

}

int main(void)

{

hMutex=CreateMutex(NULL, FALSE, NULL);

hThr=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)Thread,NULL,0,&uThrID);

while(1)

{

WaitForSingleObject(hMutex, INFINITE);

printf("%d %d %d %d %d\n", a[0], a[1], a[2], a[3], a[4]);

ReleaseMutex(hMutex);

}

return 0;

}

События

Объекты-события используются для уведомления ожидающих нитей о наступлении какого-либо события. Различают два вида событий - с ручным и автоматическим сбросом. Ручной сброс осуществляется функцией ResetEvent. События с ручным сбросом используются для уведомления сразу нескольких нитей. При использовании события с автосбросом уведомление получит и продолжит свое выполнение только одна ожидающая нить, остальные будут ожидать дальше.

Функция CreateEvent создает объект-событие, SetEvent - устанавливает событие в сигнальное состояние, ResetEvent - сбрасывает событие. Функция PulseEvent устанавливает событие, а после возобновления ожидающих это событие нитей (всех при ручном сбросе и только одной при автоматическом), сбрасывает его. Если ожидающих нитей нет, PulseEvent просто сбрасывает событие.

Пример. Синхронизация нитей с помощью событий.

 

#include <windows.h>

#include <stdio.h>

HANDLE hEvent1, hEvent2;

int a[5];

HANDLE hThr;

unsigned long uThrID;

void Thread(void* pParams)

{

int i, num = 0;

while (1)

{

WaitForSingleObject(hEvent2, INFINITE);

for (i=0; i<5; i++) a[i] = num;

num++;

SetEvent(hEvent1);

}

}

int main(void)

{

hEvent1=CreateEvent(NULL, FALSE, TRUE, NULL);

hEvent2=CreateEvent(NULL, FALSE, FALSE, NULL);

hThr=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)Thread,NULL,0,&uThrID);

while(1)

{

WaitForSingleObject(hEvent1, INFINITE);

printf("%d %d %d %d %d\n", a[0], a[1], a[2], a[3], a[4]);

SetEvent(hEvent2);

}

return 0;

}

 

Семафоры

Объект-семафор - это фактически объект-взаимоисключение со счетчиком. Данный объект позволяет "захватить" себя определенному количеству нитей. После этого "захват" будет невозможен, пока одна из ранее "захвативших" семафор нитей не освободит его. Семафоры применяются для ограничения количества нитей, одновременно работающих с ресурсом. Объекту при инициализации передается максимальное число нитей, после каждого "захвата" счетчик семафора уменьшается. Сигнальному состоянию соответствует значение счетчика больше нуля. Когда счетчик равен нулю, семафор считается не установленным (сброшенным).

Функция CreateSemaphore создает объект-семафор с указанием и максимально возможного начального его значения, OpenSemaphore – возвращает дескриптор существующего семафора, захват семафора производится с помощью ожидающих функций, при этом значение семафора уменьшается на единицу, ReleaseSemaphore - освобождение семафора с увеличением значения семафора на указанное в параметре число.

Пример. Синхронизация нитей с помощью семафоров.

 

#include <windows.h>

#include <stdio.h>

HANDLE hSem;

int a[5];

HANDLE hThr;

unsigned long uThrID;

void Thread(void* pParams)

{

int i, num = 0;

while (1)

{

WaitForSingleObject(hSem, INFINITE);

for (i=0; i<5; i++) a[i] = num;

num++;

ReleaseSemaphore(hSem, 1, NULL);

}

}

int main(void)

{

hSem=CreateSemaphore(NULL, 1, 1, "MySemaphore1");

hThr=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)Thread,NULL,0,&uThrID);

while(1)

{

WaitForSingleObject(hSem, INFINITE);

printf("%d %d %d %d %d\n", a[0], a[1], a[2], a[3], a[4]);

ReleaseSemaphore(hSem, 1, NULL);

}

return 0;

}

 




Дата добавления: 2014-12-18; просмотров: 25 | Поможем написать вашу работу | Нарушение авторских прав




lektsii.net - Лекции.Нет - 2014-2024 год. (0.011 сек.) Все материалы представленные на сайте исключительно с целью ознакомления читателями и не преследуют коммерческих целей или нарушение авторских прав