Microsoft MVP성태의 닷넷 이야기
.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점 [링크 복사], [링크+제목 복사],
조회: 16004
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 7개 있습니다.)
.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점
; https://www.sysnet.pe.kr/2/0/13156

.NET Framework: 2065. C# - Mutex의 비동기 버전
; https://www.sysnet.pe.kr/2/0/13157

닷넷: 2216. C# - SemaphoreSlim 사용 시 주의점
; https://www.sysnet.pe.kr/2/0/13555

닷넷: 2217. C# - 최댓값이 1인 SemaphoreSlim 보다 Mutex 또는 lock(obj)를 선택하는 것이 나은 이유
; https://www.sysnet.pe.kr/2/0/13558

디버깅 기술: 195. windbg 분석 사례 - Semaphore 잠금으로 인한 Hang 현상 (닷넷)
; https://www.sysnet.pe.kr/2/0/13560

닷넷: 2284. C# - async 메서드에서의 lock/Monitor.Enter/Exit 잠금 처리
; https://www.sysnet.pe.kr/2/0/13697

닷넷: 2285. C# - async 메서드에서의 System.Threading.Lock 잠금 처리
; https://www.sysnet.pe.kr/2/0/13698




C# - Mutex와 Semaphore/SemaphoreSlim 차이점

그러고 보니, 제 블로그에서 MutexSemaphore에 대해 다룬 적이 거의 없었군요. ^^; 이참에 한번 정리해보겠습니다.

두 개 모두 동기화 개체지만, 사실 Mutex는 Semaphore의 특별한 사례에 속합니다. 즉, Semaphore를 이용하면 Mutex처럼 구현하는 것도 가능합니다.

예를 들어 볼까요?

namespace mutex_semaphore
{
    internal class Program
    {
        static Mutex _m = new Mutex(false);

        static void Main(string[] args)
        {
            Task t1 = CreateNewTask(1);
            Task t2 = CreateNewTask(2);

            t1.Wait();
            t2.Wait();
            Console.WriteLine("Main-end");
        }

        private static Task CreateNewTask(int workId)
        {
            return Task.Run(() =>
            {
                _m.WaitOne();
                Console.WriteLine($"{DateTime.Now:T} [{workId}]: Sleep-before");
                Thread.Sleep(2000);
                Console.WriteLine($"{DateTime.Now:T} [{workId}]: Sleep-after");
                _m.ReleaseMutex();
            });
        }
    }
}

/* 출력 결과
오전 11:20:54 [2]: Sleep-before
오전 11:20:56 [2]: Sleep-after
오전 11:20:56 [1]: Sleep-before
오전 11:20:58 [1]: Sleep-after
Main-end
*/

위의 프로그램은 Task.Run 내부의 작업에 대해 Mutex로 WaitOne/ReleaseMutex로 동기화를 했기 때문에 단 하나의 스레드만 내부 코드를 실행할 수 있습니다.

동일한 효과를 세마포어를 이용하면 다음과 같이 구현할 수 있습니다.

namespace mutex_semaphore
{
    internal class Program
    {
        static Semaphore _smp = new Semaphore(1, 1);

        static void Main(string[] args)
        {
            // ...[생략]...
        }

        private static Task CreateNewTask(int workId)
        {
            return Task.Run(() =>
            {
                _smp.WaitOne();
                Console.WriteLine($"{DateTime.Now:T} [{workId}]: Sleep-before");
                Thread.Sleep(2000);
                Console.WriteLine($"{DateTime.Now:T} [{workId}]: Sleep-after");
                _smp.Release();
            });
        }
    }
}

/* 출력 결과
오전 11:20:05 [1]: Sleep-before
오전 11:20:07 [1]: Sleep-after
오전 11:20:07 [2]: Sleep-before
오전 11:20:09 [2]: Sleep-after
Main-end
*/

그런데, 이렇게 동일하게 구현할 수 있어도 그 둘 간의 재미있는 차이점이 하나 있습니다.




그 차이점이란, 바로 "재진입 가능성"입니다. 즉, Lock과 Thread에 결합이 있느냐가 문제가 됩니다.

우선, Mutex는 그것을 진입한 스레드, 즉 WaitOne을 호출한 스레드가 동일하게 ReleaseMutex를 호출해야 합니다. 그렇지 않으면,

static Mutex _m = new Mutex(false);

static void Main(string[] args)
{
    _m.WaitOne();
    CreateReleaseTask(1).Wait();
}

private static Task CreateReleaseTask(int workId)
{
    return Task.Run(() =>
    {
        Console.WriteLine($"{DateTime.Now:T} [{workId}]: releasing");
        try
        {
            _m.ReleaseMutex();
            Console.WriteLine($"{DateTime.Now:T} [{workId}]: released");
        }
        catch (Exception e)
        {
            Console.WriteLine(e.Message);
        }
    });
}

/* 출력 결과
오전 11:40:58 [1]: releasing
Object synchronization method was called from an unsynchronized block of code.
*/

위와 같이 ReleaseMutex에서 예외가 발생합니다. 이런 특징은 같은 스레드 내에서 WaitOne을 잠금 없이 다시 진입하는 것을 허용합니다. 대신 lock-count와 release-count 횟수를 일치시켜야만 lock이 풀립니다. 아래는 그에 대한 실행 예제를 보여줍니다.

_m.WaitOne(); // mutex lock 진입
_m.WaitOne(); // 같은 스레드에서 mutex lock 재진입 가능

_m.ReleaseMutex(); // 같은 스레드에서 release했지만, 여전히 lock 상태
Console.WriteLine($"{DateTime.Now:T} : new Task");
Task t1 = CreateNewTask(1);

Thread.Sleep(5000);
_m.ReleaseMutex(); // 같은 스레드에서 WaitOne 호출과 짝을 맞춘 Release 호출 시 비로소 lock 해제
t1.Wait();

/* 출력 결과
오전 11:44:37 : new Task
오전 11:44:42 [1]: Sleep-before
오전 11:44:44 [1]: Sleep-after
*/




반면 세마포어는 Lock과 Thread의 결합이 없습니다. 그래서, 서로 다른 스레드에서 lock을 잠그고 해제하는 것이 가능해 다음과 같은 예제가 잘 동작합니다.

_smp.WaitOne();
var t1 = CreateNewTask(1);
CreateReleaseTask(2).Wait();

t1.Wait();

private static Task CreateReleaseTask(int workId)
{
    return Task.Run(() =>
    {
        Console.WriteLine($"{DateTime.Now:T} [{workId}]: releasing");
        // ...[생략]...
        _smp.Release();
        Console.WriteLine($"{DateTime.Now:T} [{workId}]: released");
        // ...[생략]...
    });
}

/* 출력 결과
오후 12:00:21 [2]: releasing
오후 12:00:21 [2]: released
오후 12:00:21 [1]: Sleep-before
오후 12:00:23 [1]: Sleep-after
*/

보는 바와 같이 Main 스레드에서 lock을 획득 후 CreateNewTask로 해당 lock이 필요한 작업을 스레드로 시작한 다음, lock 해제를 다른 스레드에서 할 수 있어 CreateNewTask의 작업이 lock을 획득하면서 진행하고 있습니다.

이로 인해, 동일한 스레드에서 lock을 획득 시 주의해야 합니다. Mutex에서는 가능했던 다음의 코드가,

static Semaphore _smp = new Semaphore(1, 1);

static void Main(string[] args)
{
    _smp.WaitOne(); // 여기서의 잠금 한 번은 성공적으로 획득했지만,
    _smp.WaitOne(); // hang!!!! 같은 스레드에서 Semaphore lock 재진입 불가능
    Console.WriteLine("Main-end");
}

Semaphore에서는 WaitOne을 호출할 때마다 "현재 스레드가 이미 lock을 소유하고 있다는" 연관이 없으므로 매번 필요한 lock을 소유하는 식이어서, 위의 코드에서 두 번째 WaitOne은 무한 대기를 하는 현상이 나옵니다.




마지막으로, SemaphoreSlim과 Semaphore의 차이점은 뭘까요? 이에 대해서는 공식 문서에서 잘 설명하고 있습니다.

Semaphore and SemaphoreSlim
; https://learn.microsoft.com/en-us/dotnet/standard/threading/semaphore-and-semaphoreslim

그러니까, Semaphore와 Mutex는 운영체제가 제공하는 동기화 개체를 재사용하는 반면 SemaphoreSlim은 닷넷 런타임에서 스스로 구현한 개체입니다. 따라서 이름에서 유추할 수 있듯이 slim하기 때문에 성능 면에서 Semaphore보다 SemaphoreSlim이 더 낫습니다.

둘 간의 극명한 차이점은, Semaphore는 그것에 "이름"을 부여해 초기화하는 것이 가능합니다.

static Semaphore _named_smpp = new Semaphore(1, 1, "named.sem");

이렇게 주어진 이름으로, 서로 다른 프로세스 간에 해당 이름으로 동일한 Semaphore를 열 수 있습니다. 즉, inter-process 수준으로 동기화를 제공하는 것입니다.

당연히 닷넷 런타임 내에서 구현한 SemaphoreSlim은 운영체제의 도움을 받지 않으므로 프로세스 간 동기화에는 사용할 수 없습니다. (애당초 이름을 받는 생성자가 없습니다.)

그렇다면 "unnamed Semaphore"와 "SemaphoreSlim"의 차이는 뭘까요?

닷넷 프레임워크 시절에는, "다중 AppDomain"을 이용해 하나의 EXE 프로세스에서 여러 닷넷 응용 프로그램이 올라올 수 있었는데요, 그런 상황에서 "unnamed Semaphore"를 사용하면 inter-appdomain 간에 동기화가 가능했습니다. 반면 SemaphoreSlim은 AppDomain 내에서의 동기화만 가능하고.

따라서, 닷넷 코어/5+에서는 "다중 AppDomain"을 지원하지 않으므로 결국 그 둘 간의 선택 차이가 없어졌습니다. 어차피 단일 AppDomain이기 때문에, 기왕이면 SemaphoreSlim의 사용을 (성능상으로도 이점이 있으므로) 권장합니다. (달리 말해, 이제는 unnamed인 경우 Semaphore를 쓸 이유가 없어졌습니다.)

그리고 약간 혼란스럽지만, 다음과 같은 내용도 있습니다.

However, it also provides lazily initialized, kernel-based wait handles as necessary to support waiting on multiple semaphores. SemaphoreSlim also supports the use of cancellation tokens, but it does not support named semaphores or the use of a wait handle for synchronization.


그러니까, wait handle을 지원하지 않는다는 것인데 다중 세마포어의 대기를 할 때는 wait handle을 제공한다고 합니다. ^^;

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




이 정도면, 대충 Mutex와 Semaphore/SemaphoreSlim에 대한 차이점은 정리가 된 것 같습니다. 참고로, 아주 관련이 있는 것은 아니지만 다음의 글들도 읽어보시면 도움이 될 것입니다. ^^

Named 동기화 개체 생성 시 System.UnauthorizedAccessException 예외 발생하는 경우
; https://www.sysnet.pe.kr/2/0/1170

.NET 코드 - 단일 Process 실행
; https://www.sysnet.pe.kr/2/0/967




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/26/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12662정성태6/8/202115400.NET Framework: 1064. C# COM 개체를 PIA(Primary Interop Assembly)로써 "Embed Interop Types" 참조하는 방법파일 다운로드1
12661정성태6/4/202127412.NET Framework: 1063. C# - MQTT를 이용한 클라이언트/서버(Broker) 통신 예제 [4]파일 다운로드1
12660정성태6/3/202118128.NET Framework: 1062. Windows Forms - 폼 내에서 발생하는 마우스 이벤트를 자식 컨트롤 영역에 상관없이 수신하는 방법 [1]파일 다운로드1
12659정성태6/2/202119028Linux: 40. 우분투 설치 후 MBR 디스크 드라이브 여유 공간이 인식되지 않은 경우 - Logical Volume Management
12658정성태6/2/202116812Windows: 194. Microsoft Store에 있는 구글의 공식 Youtube App
12657정성태6/2/202117614Windows: 193. 윈도우 패키지 관리자 - winget 설치
12656정성태6/1/202115929.NET Framework: 1061. 서버 유형의 COM+에 적용할 수 없는 Server GC
12655정성태6/1/202114577오류 유형: 722. windbg/sos - savemodule - Fail to read memory
12654정성태5/31/202115240오류 유형: 721. Hyper-V - Saved 상태의 VM을 시작 시 오류 발생
12653정성태5/31/202118505.NET Framework: 1060. 닷넷 GC에 새롭게 구현되는 DPAD(Dynamic Promotion And Demotion for GC)
12652정성태5/31/202115948VS.NET IDE: 164. Visual Studio - Web Deploy로 Publish 시 암호창이 매번 뜨는 문제
12651정성태5/31/202116143오류 유형: 720. PostgreSQL - ERROR: 22P02: malformed array literal: "..."
12650정성태5/17/202115456기타: 82. OpenTabletDriver의 버튼에 더블 클릭을 매핑 및 게임에서의 지원 방법
12649정성태5/16/202117676.NET Framework: 1059. 세대 별 GC(Garbage Collection) 방식에서 Card table의 사용 의미 [1]
12648정성태5/16/202116431사물인터넷: 66. PC -> FTDI -> NodeMCU v1 ESP8266 기기를 UART 핀을 연결해 직렬 통신하는 방법파일 다운로드1
12647정성태5/15/202116711.NET Framework: 1058. C# - C++과의 연동을 위한 구조체의 fixed 배열 필드 사용파일 다운로드1
12646정성태5/15/202115508사물인터넷: 65. C# - Arduino IDE의 Serial Monitor 기능 구현파일 다운로드1
12645정성태5/14/202115567사물인터넷: 64. NodeMCU v1 ESP8266 - LittleFS를 이용한 와이파이 접속 정보 업데이트파일 다운로드1
12644정성태5/14/202116819오류 유형: 719. 윈도우 - 제어판의 "프로그램 및 기능" / "Windows 기능 켜기/끄기" 오류 0x800736B3
12643정성태5/14/202116757오류 유형: 718. 서버 유형의 COM+ 사용 시 0x80080005(Server execution failed) 오류 발생
12642정성태5/14/202118491오류 유형: 717. The 'Microsoft.ACE.OLEDB.12.0' provider is not registered on the local machine.
12641정성태5/13/202117305디버깅 기술: 179. 윈도우용 .NET Core 3 이상에서 Windbg의 sos 사용법
12640정성태5/13/202120898오류 유형: 716. RDP 연결 - Because of a protocol error (code: 0x112f), the remote session will be disconnected. [1]
12639정성태5/12/202117303오류 유형: 715. Arduino: Open Serial Monitor - The module '...\detection.node' was compiled against a different Node.js version using NODE_MODULE_VERSION
12638정성태5/12/202117547사물인터넷: 63. NodeMCU v1 ESP8266 - 펌웨어 내 파일 시스템(SPIFFS, LittleFS) 및 EEPROM 활용
12637정성태5/10/202117766사물인터넷: 62. NodeMCU v1 ESP8266 보드의 A0 핀에 다중 아날로그 센서 연결 [1]
... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...