Microsoft MVP성태의 닷넷 이야기
.NET Framework: 347. C# - 프로세스(EXE) 수준의 Singleton 개체 생성 [링크 복사], [링크+제목 복사],
조회: 26954
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)

C# - 프로세스(EXE) 수준의 Singleton 개체 생성

운영체제에 의해서 프로세스 격리가 제공되는 것을, 닷넷은 다시 프로세스 내부를 AppDomain이라는 단위로 격리시키고 서로 간의 직접 접근이 불가능하도록 만들었습니다.

그래서, Native 언어(예: C/C++)에서는 static으로 singleton을 만들었던 것을 닷넷에서는 static으로 지정했다고 프로세스 전역적으로 한 개만 존재하지는 않습니다, AppDomain 별로 존재하는 것이지.

이에 대한 해결책을 다음의 글에서 제공하고 있습니다.

Cross AppDomain Singleton 
; http://ingebrigtsen.info/2007/05/18/cross-appdomain-singleton/

근데, 코드가 별로 마음에 안 듭니다. ^^ Interop DLL이 별도로 추가된다는 것은 관리적인 요소만 하나 더 늘리게 되는 요인이 됩니다. 그래서, 아래의 방법을 곁들이면 더 좋습니다. ^^

레지스트리 등록 및 Interop DLL 없이 COM 개체 사용하는 방법
; https://www.sysnet.pe.kr/2/0/1180

public class CrossAppDomainSingleton<T> : MarshalByRefObject where T : new()
{
	...[생략]...
    private static AppDomain GetAppDomain(string friendlyName)
    {
        IntPtr enumHandle = IntPtr.Zero;
        ICorRuntimeHost host = null;

        try
        {
            host = Utility.CoCreateInstance("{CB2F6723-AB3A-11D2-9C40-00C04FA30A3E}") as ICorRuntimeHost;

            if (host == null)
            {
                return null;
            }

            host.EnumDomains(out enumHandle);

            object domain = null;
            while (true)
            {
                host.NextDomain(enumHandle, out domain);
                if (domain == null)
                {
                    break;
                }
                AppDomain appDomain = (AppDomain)domain;
                if (appDomain.FriendlyName.Equals(friendlyName))
                {
                    return appDomain;
                }
            }
        }
        finally
        {
            host.CloseEnum(enumHandle);
            Marshal.ReleaseComObject(host);
            host = null;
        }
        return null;
    }

	...[생략]...
}

지면(?) 관계상 ICorRuntimeHost 등의 구현은 이 글의 첨부 파일로 참고하시면 됩니다.

그나저나, 어쩔 수 없이 프로세스 수준의 Singleton을 사용해야 하지만 성능 손실은 과연 어느 정도일까요? 이를 위해 다음과 같이 테스트 코드를 작성해 봤습니다.

public static class NonSingleton
{
    static bool _test = true;
    public static bool Test { get { return _test; } }
}

public class Singleton : CrossAppDomainSingleton<Singleton>
{
    bool _test = true;
    public bool Test { get { return _test; } }
}

class Program
{
    static void Main(string[] args)
    {
        CheckTime(false, 1); // JIT 컴파일 시간을 배제하기 위해.
        CheckTime(true, 1000000);
    }

    private static void CheckTime(bool outputResult, int loopCount)
    {
        Stopwatch st = new Stopwatch();
        st.Start();

        int inc = 0;
        for (int i = 0; i < loopCount; i++)
        {
            if (NonSingleton.Test == true)
            {
                inc++;
            }
        }
        st.Stop();

        if (outputResult == true)
        {
            Console.WriteLine(st.ElapsedMilliseconds);
        }

        System.Diagnostics.Trace.WriteLine(inc);

        st.Start();

        inc = 0;
        Singleton singleton = Singleton.Instance;
        for (int i = 0; i < loopCount; i++)
        {
            if (singleton.Test == true)
            {
                inc++;
            }
        }

        st.Stop();

        if (outputResult == true)
        {
            Console.WriteLine(st.ElapsedMilliseconds);
        }

        System.Diagnostics.Trace.WriteLine(inc);
    }
}

Release 빌드 후 테스트 결과는 다음과 같이 나왔습니다.

10,000 번 호출 NonSingleton: 0ms
10,000 번 호출 Singleton: 4ms

100,000 번 호출 NonSingleton: 0ms
100,000 번 호출 Singleton: 45ms

1,000,000 번 호출 NonSingleton: 1ms
1,000,000 번 호출 Singleton: 454ms

10,000,000 번 호출 NonSingleton: 19ms
10,000,000 번 호출 Singleton: 4539ms

거의 200배가 넘게 느린 결과를 볼 수 있는데요. 무척 느리다 싶지만 이런 유의 테스트가 현실로 왔을 때는 재미있게 해석될 수 있습니다. 예를 들어, 1초에 10,000번 호출된다고 가정했을 때 도메인간 호출이 겨우 4ms 더 시간이 걸릴 뿐이므로 그다지 나쁜 결과는 아니라는 것입니다.

** 주의할 점은, 이는 호출 간의 속도 저하일 뿐 마샬링이 복잡해지면 더욱 느려질 수 있다는 사실!!!




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 2/27/2024]

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

비밀번호

댓글 작성자
 



2012-12-06 04시26분
[ryujh] 안녕하세요.

프로세스를 중복실행 방지하기 위해 뮤텍스를 사용하는데 이것이 Singleton 으로 봐도 될지요?

위의 예제는 해본적이 없어서 좀더 보고 질문드리겠습니다.
[guest]
2012-12-06 04시47분
개체가 하나면 싱글턴 개체라고 하는 것처럼, 프로세스가 하나라면 싱글턴 프로세스라는 말을 써도 되겠지요. 싱글턴 프로세스라면 위의 기법을 사용하기보다는 말씀하신 것처럼 뮤텍스가 올바른 선택일 수 있습니다.
정성태

... 16  17  18  19  20  21  22  23  [24]  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13337정성태5/5/202312841Linux: 59. dockerfile - docker exec로 container에 접속 시 자동으로 실행되는 코드 적용
13336정성태5/4/202312365.NET Framework: 2111. C# - 바이너리 출력 디렉터리와 연관된 csproj 설정
13335정성태4/30/202313277.NET Framework: 2110. C# - FFmpeg.AutoGen 라이브러리를 이용한 기본 프로젝트 구성 - Windows Forms파일 다운로드1
13334정성태4/29/202312280Windows: 250. Win32 C/C++ - Modal 메시지 루프 내에서 SetWindowsHookEx를 이용한 Thread 메시지 처리 방법
13333정성태4/28/202310806Windows: 249. Win32 C/C++ - 대화창 템플릿을 런타임에 코딩해서 사용파일 다운로드1
13332정성태4/27/202311101Windows: 248. Win32 C/C++ - 대화창을 위한 메시지 루프 사용자 정의파일 다운로드1
13331정성태4/27/202310726오류 유형: 856. dockerfile - 구 버전의 .NET Core 이미지 사용 시 apt update 오류
13330정성태4/26/202311289Windows: 247. Win32 C/C++ - CS_GLOBALCLASS 설명
13329정성태4/24/202311472Windows: 246. Win32 C/C++ - 직접 띄운 대화창 템플릿을 위한 Modal 메시지 루프 생성파일 다운로드1
13328정성태4/19/202311297VS.NET IDE: 184. Visual Studio - Fine Code Coverage에서 동작하지 않는 Fake/Shim 테스트
13327정성태4/19/202311505VS.NET IDE: 183. C# - .NET Core/5+ 환경에서 Fakes를 이용한 단위 테스트 방법
13326정성태4/18/202314080.NET Framework: 2109. C# - 닷넷 응용 프로그램에서 SQLite 사용 (System.Data.SQLite) [1]파일 다운로드1
13325정성태4/18/202312235스크립트: 48. 파이썬 - PostgreSQL의 with 문을 사용한 경우 연결 개체 누수
13324정성태4/17/202312480.NET Framework: 2108. C# - Octave의 "save -binary ..."로 생성한 바이너리 파일 분석파일 다운로드1
13323정성태4/16/202312129개발 환경 구성: 677. Octave에서 Excel read/write를 위한 io 패키지 설치
13322정성태4/15/202313661VS.NET IDE: 182. Visual Studio - 32비트로만 빌드된 ActiveX와 작업해야 한다면?
13321정성태4/14/202311576개발 환경 구성: 676. WSL/Linux Octave - Python 스크립트 연동
13320정성태4/13/202311349개발 환경 구성: 675. Windows Octave 8.1.0 - Python 스크립트 연동
13319정성태4/12/202312322개발 환경 구성: 674. WSL 2 환경에서 GNU Octave 설치
13318정성태4/11/202312000개발 환경 구성: 673. JetBrains IDE에서 "Squash Commits..." 메뉴가 비활성화된 경우
13317정성태4/11/202312195오류 유형: 855. WSL 2 Ubuntu 20.04 - error: cannot communicate with server: Post http://localhost/v2/snaps/...
13316정성태4/10/202310674오류 유형: 854. docker-compose 시 "json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)" 오류 발생
13315정성태4/10/202311610Windows: 245. Win32 - 시간 만료를 갖는 컨텍스트 메뉴와 윈도우 메시지의 영역별 정의파일 다운로드1
13314정성태4/9/202312472개발 환경 구성: 672. DosBox를 이용한 Turbo C, Windows 3.1 설치 [1]
13313정성태4/9/202311888개발 환경 구성: 671. Hyper-V VM에 Turbo C 2.0 설치 [2]
13312정성태4/8/202311632Windows: 244. Win32 - 시간 만료를 갖는 MessageBox 대화창 구현 (개선된 버전)파일 다운로드1
... 16  17  18  19  20  21  22  23  [24]  25  26  27  28  29  30  ...