Форум программистов, компьютерный форум CyberForum.ru

mutex - C++

Восстановить пароль Регистрация
 
Рейтинг: Рейтинг темы: голосов - 15, средняя оценка - 4.67
cppGhost
32 / 30 / 1
Регистрация: 21.06.2012
Сообщений: 91
27.06.2012, 20:20     mutex #1
День добрый. Прошу объяснить мне дураку
C++
1
2
3
4
5
m_hShared = OpenMutex(MUTEX_ALL_ACCESS, TRUE, L"FileMutexEx");
if(m_hShared)
    WaitForSingleObject(m_hShared, INFINITE);                    
        
m_hShared = CreateMutex(NULL, TRUE, L"FileMutexEx");
Если два раза из одного и того же потока вызвать этот код, то проблема следующая. Во второй раз при вызове CreateMutex возвращается новый хендл, однако происходит ошибка, в которой говорится,
что мьютекс уже существует. То есть получается, что создать не получилось, но какой-то хендл возвращается. Как так?
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
fit
14 / 14 / 0
Регистрация: 20.04.2010
Сообщений: 102
27.06.2012, 20:34     mutex #2
мб потому что объект с тем же именем создается?
cppGhost
32 / 30 / 1
Регистрация: 21.06.2012
Сообщений: 91
27.06.2012, 20:36  [ТС]     mutex #3
GetLastError возвращает ошибку, в документации сказано, что если такой есть, то должен вернуться старый дескриптор
fit
14 / 14 / 0
Регистрация: 20.04.2010
Сообщений: 102
27.06.2012, 20:49     mutex #4
странно. повторное создание именнованных объектов ядра в том же пространстве имен всегда завершается ошибкой. 100%
cppGhost
32 / 30 / 1
Регистрация: 21.06.2012
Сообщений: 91
27.06.2012, 20:53  [ТС]     mutex #5
Так и завершается с ошибкой, то есть это нормально я у же понял. Я не пойму, почему возвращается новый хендл. Может это у меня с компом что-то?
fit
14 / 14 / 0
Регистрация: 20.04.2010
Сообщений: 102
27.06.2012, 20:54     mutex #6
не знаю что вам нужно, но мб DuplicateHandle было бы уместее использовать вместо повторного создания мьютекса
cppGhost
32 / 30 / 1
Регистрация: 21.06.2012
Сообщений: 91
27.06.2012, 21:01  [ТС]     mutex #7
Хорошо, объясню, что именно я пробую. На самом деле просто изучаю мьютексы =))
Я запускаю два экземпляра одной программы. Там две кнопки, "взять" и "освободить". Как я думал, что будет работать: в первой программе нажимаю "взять", потом во второй "Взять". Вторая программа зависает в ожидании. Потом в первой я нажимаю "освободить" и при этом вторая начинает работать. Это получается.

Проблема в том, что если сделаю "взять_1", "взять_2", снова "взять_1" то вторая программа уже никогда не получает управление. Может я просто неправильно организовываю синхронизацию?
fit
14 / 14 / 0
Регистрация: 20.04.2010
Сообщений: 102
27.06.2012, 21:02     mutex #8
Цитата Сообщение от cppGhost Посмотреть сообщение
Так и завершается с ошибкой, то есть это нормально я у же понял. Я не пойму, почему возвращается новый хендл. Может это у меня с компом что-то?
с чего вы взяли, что новый описатель вернулся? этот как раз по коду GetLastError определяется вроде. а оно у вас ошибку возвращает.
ради интереса попробуйте с мьютексом без имени повторное создание проделать, может и вернет старый. только толку-то)
cppGhost
32 / 30 / 1
Регистрация: 21.06.2012
Сообщений: 91
27.06.2012, 21:05  [ТС]     mutex #9
Новый описатель, так как в дебаге я вижу, что он новый =))
fit
14 / 14 / 0
Регистрация: 20.04.2010
Сообщений: 102
27.06.2012, 21:13     mutex #10
Цитата Сообщение от cppGhost Посмотреть сообщение
Хорошо, объясню, что именно я пробую. На самом деле просто изучаю мьютексы =))
Я запускаю два экземпляра одной программы. Там две кнопки, "взять" и "освободить". Как я думал, что будет работать: в первой программе нажимаю "взять", потом во второй "Взять". Вторая программа зависает в ожидании. Потом в первой я нажимаю "освободить" и при этом вторая начинает работать. Это получается.

Проблема в том, что если сделаю "взять_1", "взять_2", снова "взять_1" то вторая программа уже никогда не получает управление. Может я просто неправильно организовываю синхронизацию?
когда вы нажимаете "взять 1", а потом "взять 2" вторая программа переходит в состояние ожидания (так как мьютекс уже занят, его идентификатор содержит идентификатор потока 1ой программы, а взятие мьютекса можно провети только если мьютекс свободен, т.е. если его ид=0).
далее вы нажимаете еще раз "взять 1" и 1ая программа второй раз занимает мьютекс (счетчик рекурсии мьютекса равен 2).
очевидно, что в такой ситуации никакой поток не займет мьютекс пока его не освободит поток 1ой программы 2 раза

Добавлено через 1 минуту
Цитата Сообщение от cppGhost Посмотреть сообщение
Новый описатель, так как в дебаге я вижу, что он новый =))
это все фигня если GetLastError не вернул код, что вернулся новый описатель, значит он не вернулся
cppGhost
32 / 30 / 1
Регистрация: 21.06.2012
Сообщений: 91
27.06.2012, 21:30  [ТС]     mutex #11
По поводу счетчика навели меня на мысль и она оказалась правильной (похоже)

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
void CMutexDlg::CreateMutex()
{
    m_hShared = OpenMutex(MUTEX_ALL_ACCESS, FALSE, L"FileMutexEx");
    if(m_hShared)
        WaitForSingleObject(m_hShared, INFINITE);                   
            
    m_handles.push_back(CreateMutex(NULL, TRUE, L"FileMutexEx"));
    AfxMessageBox(L"Ресурс залочен");
}
 
void CMutexDlg::ReleaseMutex()
{
    if(m_handles.size() > 0)
    {
        HANDLE h = m_handles[m_handles.size()-1];
        ReleaseMutex(h);
        CloseHandle(h);
        m_handles.erase(m_handles.begin() + m_handles.size() - 1);
        
        AfxMessageBox(L"ресурс освобожден");
    }
}
то есть сначала "создаем мьютекс" N раз, и потом столько же освобождаем. единственное меня все-таки смущает такая реализация
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
27.06.2012, 21:57     mutex
Еще ссылки по теме:

C++ Можно ли обойтись без mutex?
C++ Std::mutex выдает исключение abort() при попытки unlock()
Std::atomic vs std::mutex C++

Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
fit
14 / 14 / 0
Регистрация: 20.04.2010
Сообщений: 102
27.06.2012, 21:57     mutex #12
ну если надо несколько раз мьютекс занять он не создается, а захватывается через Wait функцию. по другому как-то странно.
так работает мьютекс, с этим ничего не поделать. это кстати, единственный объект ядра с которым срабатывает фишка с несколькими захватами. при работе с другими объектами (например тут можно поставить семафор) повторно захватить их несколько раз одним и тем же потоком не получится
Yandex
Объявления
27.06.2012, 21:57     mutex
Ответ Создать тему
Опции темы

Текущее время: 08:03. Часовой пояс GMT +3.
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2016, vBulletin Solutions, Inc.
Рейтинг@Mail.ru