Microsoft MVP성태의 닷넷 이야기
.NET Framework: 347. C# - 프로세스(EXE) 수준의 Singleton 개체 생성 [링크 복사], [링크+제목 복사],
조회: 28406
글쓴 사람
정성태 (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분
개체가 하나면 싱글턴 개체라고 하는 것처럼, 프로세스가 하나라면 싱글턴 프로세스라는 말을 써도 되겠지요. 싱글턴 프로세스라면 위의 기법을 사용하기보다는 말씀하신 것처럼 뮤텍스가 올바른 선택일 수 있습니다.
정성태

... 136  137  138  139  [140]  141  142  143  144  145  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1585정성태1/14/201431471VS.NET IDE: 84. Visual Studio를 이용한 파일 비교(diff)
1584정성태1/13/201433558Windows: 83. 윈도우 8 - UI가 있는 프로그램을 Local SYSTEM 권한의 세션 0 데스크톱에서 실행하는 방법
1583정성태1/13/201431915Windows: 82. 윈도우 8 - "Interactive Services Detection" 서비스 시작하는 방법 [1]
1582정성태1/12/201430177개발 환경 구성: 210. 원격 데스크톱(RDP) 접속 프로그램 - Royal TS [1]
1581정성태1/12/201431569.NET Framework: 408. 자바와 닷넷의 제네릭 차이점 - 중간 언어 및 공변/반공변 처리 [8]
1580정성태1/12/201441584.NET Framework: 407. 닷넷 사용자 정의 예외 클래스의 최소 구현 코드 [1]
1579정성태1/12/201423447오류 유형: 207. System.ArgumentException was unhandled - Message=[net_WebHeaderInvalidControlChars]
1578정성태1/11/201435224개발 환경 구성: 209. Fiddler에서 WebSocket 통신을 모니터링하는 방법 [1]
1577정성태1/11/201425246오류 유형: 206. WriteFile Win32API 사용 시 비정상 종료 현상 [3]
1576정성태1/11/201443207Windows: 81. 긴 이름의 파일/폴더 삭제하는 법 [5]
1575정성태1/11/201423246오류 유형: 205. Exception calling "Provision" with "0" argument(s): "The timer job did not complete running within the allotted time."
1574정성태1/11/201425506오류 유형: 204. An unhandled exception ('System.Security.Cryptography.CryptographicException') occurred in OWSTIMER.EXE
1573정성태1/11/201423503오류 유형: 203. 이벤트 로그 에러 - MsiInstaller에서 Chart Controls 설치 관련 오류 발생
1572정성태1/9/201426940.NET Framework: 406. CoreLab - OraDirect .NET 사용법
1571정성태1/9/201422988.NET Framework: 405. override 메서드가 정의된 타입의 인스턴스로 base 메서드를 호출하는 방법 - 두 번째 이야기
1570정성태1/9/201426581Math: 11. C# 시뮬레이션 - 몬티홀 게임파일 다운로드1
1569정성태1/8/201423590Windows: 80. 윈도우 서버 백업의 복원하는 옵션 설명
1568정성태1/8/201424324오류 유형: 202. Hyper-V 서비스 시작 오류 - Not enough storage is available to complete this operation.
1567정성태1/8/201493827기타: 40. 오피스 2013 라이선스 리셋하는 방법 [5]
1566정성태1/7/201424692Windows: 79. 윈도우 8 - 스카이드라이브(skydrive)의 Online/Offline 동기화 방식 [1]
1565정성태1/3/201423197.NET Framework: 404. 리플렉션을 이용해 닷넷 LicenseManager를 우회할 수 있는 사례
1564정성태1/3/201423706.NET Framework: 403. override 메서드가 정의된 타입의 인스턴스로 base 메서드를 호출하는 방법파일 다운로드1
1563정성태1/3/201429469오류 유형: 201. ASP.NET 웹 사이트를 IIS 7 이상의 환경에서 호스팅할 때 500 오류 발생
1562정성태1/2/201434163.NET Framework: 402. 카카오톡 PC 버전 실행 시 개발자 컴퓨터에서 Themida 오류 나는 문제 - 두 번째 이야기
1561정성태1/1/201443822오류 유형: 200. 카카오톡 PC 버전 실행 시 개발자 컴퓨터에서 Themida 오류 나는 문제 [2]
1560정성태1/1/201422335오류 유형: 199. Hyper-V - Checkpoint 생성 오류
... 136  137  138  139  [140]  141  142  143  144  145  146  147  148  149  150  ...