Microsoft MVP성태의 닷넷 이야기
.NET Framework: 630. C# - Interlocked.CompareExchange 사용 예제 [링크 복사], [링크+제목 복사],
조회: 16367
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

C# - Interlocked.CompareExchange 사용 예제

한 번만 실행하려고 하는 경우 간단하게는 다음과 같은 처리를 할 수 있습니다.

int _value = 0;
bool _done = false;

void func()
{
    if (_done == false)
    {
        _value++;
        _done = true;
    }
}

당연하지만, 이는 thread-safe하지 않기 때문에 다중 스레드에서 위의 func을 호출하려면 예기치 않은 동작이 발생할 수 있습니다. 위의 상황에서는 _value 값이 2 이상이 나올 수 있는데 이는 다음과 같은 코드로 테스트해 볼 수 있습니다.

using System;
using System.Collections.Generic;
using System.Threading;

class Program
{
    static int _value = 0;
    static bool _done = false;

    static void Main(string[] args)
    {
        int count = 0;

        while (true)
        {
            _value = 0;
            _done = false;

            List<Thread> list = new List<Thread>();
            for (int i = 0; i < 100; i++)
            {
                Thread t = new Thread(threadNotSafeFunc);
                list.Add(t);
            }

            foreach (var item in list)
            {
                item.Start();
            }

            foreach (var item in list)
            {
                item.Join();
            }

            if (_value != 1)
            {
                throw new ApplicationException("_value: " + _value);
            }

            Console.WriteLine(count++ + ": Tested - " + _value);
        }
    }

    private static void threadNotSafeFunc()
    {
        Thread.Sleep(1000);

        if (_done == false)
        {
            _value++;
            _done = true;
        }
    }
}

실행해 보면, 여지없이 ApplicationException 예외가 발생합니다. 이를 위해 lock 구문을 써도 되는데, 쓸데없이 참조 형 변수를 하나 둬야 하므로 이럴 땐 Interlocked.CompareExchange를 써 볼 수 있습니다. 원래는 다음과 같이 코딩할 수 있을 텐데,

int _value = 0;
bool _done = false;

void threadSafeFunc()
{
    if (Interlocked.CompareExchange(ref _done, true, false) == false)
    {
        _value++;
    }
}

아쉽게도 위의 코드를 컴파일하면 다음과 같은 오류가 발생합니다.

Error CS0452 The type 'bool' must be a reference type in order to use it as parameter 'T' in the generic type or method 'Interlocked.CompareExchange<T>(ref T, T, T)

bool 타입을 지원하지 않는 것인데, 사실 좀 이해가 안 됩니다. 어차피 CPU의 워드 타입 내에서 처리할 수 있는 형식이라서 .NET BCL 측에서 제공해줘도 무방할 텐데 굳이 뺀 이유를 모르겠습니다. 이 때문에, bool 타입은 그냥 true를 1로, false를 0으로 해석하는 전통적인 방식에 기인해서 int 타입을 써 다음과 같이 처리해 주시면 됩니다.

int _value = 0;
int _intDone = false;

void threadSafeFunc()
{
    if (Interlocked.CompareExchange(ref _intDone, 1, 0) == 0)
    {
        _value++;
    }
}

ApplicationException이 발생했던 이전 예제의 threadNotSafeFunc을 threadSafeFunc으로 변경해서 처리하면 예외 없이 잘 실행되는 것을 확인할 수 있습니다.

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




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







[최초 등록일: ]
[최종 수정일: 1/9/2017]

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

비밀번호

댓글 작성자
 



2017-01-10 04시56분
[spowner] 글 감사드립니다
[guest]
2017-01-10 08시04분
[spowner] lock 버젼으로... 한번 이렇게 짜봤습니다 -_-;

    class Program
    {
        static void Main(string[] args)
        {
            Flag doneFlag = false;

            doneFlag.RunOnce(() =>
            {
                Console.WriteLine("Yeah!");
            });
        }
    }

    public struct Flag
    {
        private bool flag;
        private object @lock;

        public Flag(bool flag)
        {
            this.flag = flag;
            @lock = new object();
        }

        public static implicit operator bool(Flag v)
        {
            return v.flag;
        }

        public static implicit operator Flag(bool v)
        {
            return new Flag(v);
        }

        public void RunOnce(Action action)
        {
            lock (@lock)
            {
                if (flag == true)
                    return;

                flag = true;
            }

            action();
        }
    }
[guest]
2017-01-10 12시40분
@spowner 재미있게 만드셨네요. ^^
정성태

... 61  62  [63]  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12055정성태11/17/20199254개발 환경 구성: 463. Visual Studio의 Ctrl + Alt + M, 1 (Memory 1) 등의 단축키가 동작하지 않는 경우
12054정성태11/15/201910581.NET Framework: 869. C# - 일부러 GC Heap을 깨뜨려 GC 수행 시 비정상 종료시키는 예제
12053정성태11/15/201912223Windows: 166. 윈도우 10 - 명령행 창(cmd.exe) 속성에 (DotumChe, GulimChe, GungsuhChe 등의) 한글 폰트가 없는 경우
12052정성태11/15/201911299오류 유형: 578. Azure - 일정(schedule)에 등록한 runbook이 1년 후 실행이 안 되는 문제(Reason - The key used is expired.)
12051정성태11/14/201913813개발 환경 구성: 462. 시작하자마자 비정상 종료하는 프로세스의 메모리 덤프 - procdump [1]
12050정성태11/14/201911505Windows: 165. AcLayers의 API 후킹과 FaultTolerantHeap
12049정성태11/13/201911557.NET Framework: 868. (닷넷 프로세스를 대상으로) 디버거 방식이 아닌 CLR Profiler를 이용해 procdump.exe 기능 구현
12048정성태11/12/201912389Windows: 164. GUID 이름의 볼륨에 해당하는 파티션을 찾는 방법
12047정성태11/12/201914230Windows: 163. 안전하게 eject시킨 USB 장치를 물리적인 재연결 없이 다시 인식시키는 방법
12046정성태10/29/201910199오류 유형: 577. windbg - The call to LoadLibrary(...\sos.dll) failed, Win32 error 0n193
12045정성태10/27/20199540오류 유형: 576. mstest.exe 실행 시 "Visual Studio Enterprise is required to execute the test." 오류 - 두 번째 이야기
12044정성태10/27/20199776오류 유형: 575. mstest.exe - System.Resources.MissingSatelliteAssemblyException: The satellite assembly named "Microsoft.VisualStudio.ProductKeyDialog.resources.dll, ..."
12043정성태10/27/201910531오류 유형: 574. Windows 10 설치 시 오류 - 0xC1900101 - 0x4001E
12042정성태10/26/201910915오류 유형: 573. OneDrive 하위에 위치한 Documents, Desktop 폴더에 대한 권한 변경 시 "Unable to display current owner"
12041정성태10/23/201910935오류 유형: 572. mstest.exe - The load test results database could not be opened.
12040정성태10/23/201911198오류 유형: 571. Unhandled Exception: System.Net.Mail.SmtpException: Transaction failed. The server response was: 5.2.0 STOREDRV.Submission.Exception:SendAsDeniedException.MapiExceptionSendAsDenied
12039정성태10/22/20199641스크립트: 16. cmd.exe의 for 문에서는 ERRORLEVEL이 설정되지 않는 문제
12038정성태10/17/20199215오류 유형: 570. SQL Server 2019 RC1 - SQL Client Connectivity SDK 설치 오류
12037정성태10/15/201914827.NET Framework: 867. C# - Encoding.Default 값을 바꿀 수 있을까요?파일 다운로드1
12036정성태10/14/201916128.NET Framework: 866. C# - 고성능이 필요한 환경에서 GC가 발생하지 않는 네이티브 힙 사용파일 다운로드1
12035정성태10/13/201912275개발 환경 구성: 461. C# 8.0의 #nulable 관련 특성을 .NET Framework 프로젝트에서 사용하는 방법 [2]파일 다운로드1
12034정성태10/12/201911660개발 환경 구성: 460. .NET Core 환경에서 (프로젝트가 아닌) C# 코드 파일을 입력으로 컴파일하는 방법 [1]
12033정성태10/11/201915290개발 환경 구성: 459. .NET Framework 프로젝트에서 C# 8.0/9.0 컴파일러를 사용하는 방법
12032정성태10/8/201911806.NET Framework: 865. .NET Core 2.2/3.0 웹 프로젝트를 IIS에서 호스팅(Inproc, out-of-proc)하는 방법 - AspNetCoreModuleV2 소개
12031정성태10/7/20199273오류 유형: 569. Azure Site Extension 업그레이드 시 "System.IO.IOException: There is not enough space on the disk" 예외 발생
12030정성태10/5/201915452.NET Framework: 864. .NET Conf 2019 Korea - "닷넷 17년의 변화 정리 및 닷넷 코어 3.0" 발표 자료 [1]파일 다운로드1
... 61  62  [63]  64  65  66  67  68  69  70  71  72  73  74  75  ...