Время от времени у программистов возникает вопрос как предотвратить повторный запуск программы. Это, конечно, не такой "вечный" вопрос, как проблема помещения иконки "туда, где часики", но возникает он регулярно. Предлагаются самые разные решения: от поиска уже созданного окна приложения до файлов-меток и использования PrevInstance. Каждый из этих методов имеет свои недостатки. Предлагаю почти идеальный метод распознавания уже запущенного приложения - использование семафора. Способ прост и лаконичен. Он позволяет распознать, что приложение уже запущено. А дальше все зависит от Вас: хотите не допустить повторный запуск - гасите вторую копию. А может быть нужно, чтобы некие действия выполнялись только первым экземпляром приложения - да нет проблем! И при этом не требуется громоздкого перечисления окон, не нужны файлы... В общем, сплошные плюсы. Что же такое семафор? Это объект операционной системы, предназначенный для совместного управления различными ресурсами ОС. Семафор может находиться в двух состояниях: "свободен" и "занят". С семафором связан счетчик ресурсов. Для занятого семафора он имеет нулевое значение, для свободного - положительное. Семафор можно использовать совместно с функцией WaitForSingleObject для ожидания освобождения семафора. Но нам не требуется ждать освобождения семафора. Достаточно попытаться создать семафор функциейCreateSemaphore, а затем просто проанализировать код ошибки. Если семафор создан успешно - запускается первый экземпляр. Если возникла ошибка - экземпляр не первый. И это все. Осталось только не забыть освободить семафор вызовом функции ReleaseSemaphore перед завершением приложения. Однако, к делу! Создаем VB-проект - обычный Exe-файл. Добавляем форму. Сажаем на форму кнопку, пишем на ней "Закрыть". Задаем обработчик нажатия кнопки, а также обработчик события QueryUnload для формы: |
Private Sub Command1_Click() If (semHNDL <> 0) Then RC& = ReleaseSemaphore(semHNDL, 1, CC&) End End Sub Private Sub Form_QueryUnload(Cancel As Integer, UnloadMode As Integer) If (semHNDL <> 0) Then RC& = ReleaseSemaphore(semHNDL, 1, CC&) Cancel = 0 End Sub |
Добавляем к проекту модуль. В область кода заносим: |
Public Declare Function CreateSemaphore Lib "kernel32" _ Alias "CreateSemaphoreA" _ (ByVal lpSemaphoreAttributes As Long, _ ByVal lInitialCount As Long, _ ByVal lMaximumCount As Long, _ ByVal lpName As String) As Long Public Declare Function ReleaseSemaphore Lib "kernel32" _ (ByVal hSemaphore As Long, _ ByVal lReleaseCount As Long, _ lpPreviousCount As Long) As Long Global semHNDL As Long Sub Main() Err.Clear semHNDL = CreateSemaphore(0, 0, 1, "Cats_Tail") RC& = Err.LastDllError If (RC& <> 0) Or (semHNDL = 0) Then '::: Это - не первый экземпляр MsgBox "Повторный запуск!" End End If Form1.Show End Sub |
Для нашего проекта устанавливаем startup-объект - "SUB MAIN". Транслируем. Если теперь запустить полученный
exe-файл, появится наша формочка с кнопкой "Закрыть". А теперь попробуйте запустить второй экземпляр exe-файла.
Как Вы думаете, что произойдет? Правильно: выйдет сообщение "Повторный запуск". Вы можете копировать exe-файл
в любую директорию (в т.ч. и в сеть). Запустить второй экземпляр все равно не удастся. |
Файфель Б.Л. (Cats_Tail@mail.ru) |