Microsoft MVP성태의 닷넷 이야기
.NET Framework: 106. WCF - 다중 서비스 호스트 [링크 복사], [링크+제목 복사],
조회: 17170
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일


WCF - 다중 서비스 호스트


가끔, 이에 대해서 물어보는 경우가 있어서 차후 반복되는 답변을 위해서 미리 써볼까 합니다. (사실, 그다지 기술적인 팁은 아니지만.)

먼저, 대강 WCF 서비스를 만드는 방법을 훑어 볼까요!

WCF 서비스를 위해서는 대체로 2개의 클래스/인터페이스가 필요합니다. 먼저, 아래와 같이 ServiceContract를 만들고,

[ServiceContract]
public interface IServiceA
{
    [OperationContract]
    string GetServiceName();
}

그다음, 당연히 이에 대한 구현 클래스를 만들어야죠.

[ServiceBehavior]
public class ServiceAImp : IServiceA
{
    public string GetServiceName()
    {
        return "ServiceA";
    }
}

이렇게 마련되면, 다음과 같이 서비스 호스팅을 하게 되지요.

public partial class Form1 : Form
{
    ServiceHost aHost;

    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        aHost = new ServiceHost(typeof(ServiceAImp));
        aHost.Open();
    }

    private void Form1_FormClosed(object sender, FormClosedEventArgs e)
    {
        aHost.Close();
    }
}




일단, 서비스 하나는 위와 같이 정의를 해서 호스팅을 하는 상태에서, 별도로 다른 서비스를 정의하려면 역시나 다음과 같이 ServiceContract 클래스와 ServiceBehavior 클래스를 추가해 주는 것으로 시작할 수 있습니다.

[ServiceContract]
public interface IServiceB
{
    [OperationContract]
    string GetServiceName();
}

[ServiceBehavior]
public class ServiceBImp : IServiceB
{
    public string GetServiceName()
    {
        return "ServiceB";
    }
}

질문을 하시는 분들 중에 대부분이 위와 같은 정도로 정의를 해놓고 더 이상 어찌할 바를 모릅니다. 그렇죠! 방법은 의외로 간단합니다. 별도의 ServiceHost를 하나 더 정의해서 추가해 주면 되는 것입니다.

public partial class Form1 : Form
{
    ServiceHost aHost;
    ServiceHost bHost;

    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        aHost = new ServiceHost(typeof(ServiceAImp));
        aHost.Open();
        bHost = new ServiceHost(typeof(ServiceBImp));
        bHost.Open();
    }

    private void Form1_FormClosed(object sender, FormClosedEventArgs e)
    {
        aHost.Close();
        bHost.Close();
    }
}

간단하지요. ^^

물론, app.config에 추가되는 서비스 관련 환경 설정도 ServiceA에서 해주는 것과 동일하게 ServiceB도 추가해 주면 됩니다. 물론, 서비스끼리 포트를 공유해서 설정하는 것도 가능합니다.

<system.serviceModel>
    <services>
        <!-- Service A -->
        <service behaviorConfiguration="ServiceAHost.ServiceBehavior"
                name="ServiceHostWinApp.ServiceAImp">
            <host>
                <baseAddresses>
                    <add baseAddress="http://localhost:24000/ServiceA" />
                    <add baseAddress="net.tcp://localhost:24001/" />
                </baseAddresses>
            </host>
            <endpoint address="ServiceA" binding="netTcpBinding" bindingConfiguration="ServiceA_TcpBinding"
                      contract="ServiceHostWinApp.IServiceA" />
            <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
        </service>

        <!-- Service B -->
        <service behaviorConfiguration="ServiceBHost.ServiceBehavior"
                name="ServiceHostWinApp.ServiceBImp">
            <host>
                <baseAddresses>
                    <add baseAddress="http://localhost:24000/ServiceB" />
                    <add baseAddress="net.tcp://localhost:24001/" />
                </baseAddresses>
            </host>
            <endpoint address="ServiceB" binding="netTcpBinding" bindingConfiguration="ServiceB_TcpBinding"
                      contract="ServiceHostWinApp.IServiceB" />
            <endpoint address="mex" binding="mexHttpBinding" contract="IMetadataExchange" />
        </service>
    </services>
    
    .... [중간 생략] ....
</system.serviceModel>

baseAddress 설정에서 http 부분에 대해서는 별도로 "ServiceA", "ServiceB"를 붙였는데요. 왜냐하면, mex 기능이 해당 baseAddress를 기준으로 그대로 "mex"만을 붙여서 endpoint를 정하기 때문입니다. 그래서 2개의 endpoint가 "http://localhost:24000/mex"라고 겹치기 때문에 중간에 그것을 구분해주는 서비스 명을 넣은 것입니다.

반면에, net.tcp에서는 그냥 "net.tcp://localhost:24001/"에서 마무리 지었지요. 그 대신, 각각의 서비스마다 "endpoint" 설정을 두어서 address에 "ServiceA", "ServiceB"를 추가해 놓은 것입니다. 만약 endpoint의 address 값을 공백 문자열로 주었다면 baseAddress에서부터 서비스를 구분하는 문자열을 두어야 합니다.

마지막으로, 첨부된 파일은 위의 내용들을 반영한 간단한 예제 프로젝트입니다.



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







[최초 등록일: ]
[최종 수정일: 6/28/2021]

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

비밀번호

댓글 작성자
 




... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13171정성태11/25/20225055Windows: 214. 윈도우 - 스레드 스택의 "red zone"
13170정성태11/24/20225334Windows: 213. 윈도우 - 싱글 스레드는 컨텍스트 스위칭이 없을까요?
13169정성태11/23/20225902Windows: 212. 윈도우의 Protected Process (Light) 보안 [1]파일 다운로드2
13168정성태11/22/20225252제니퍼 .NET: 31. 제니퍼 닷넷 적용 사례 (9) - DB 서비스에 부하가 걸렸다?!
13167정성태11/21/20225273.NET Framework: 2070. .NET 7 - Console.ReadKey와 리눅스의 터미널 타입
13166정성태11/20/20224995개발 환경 구성: 651. Windows 사용자 경험으로 WSL 환경에 dotnet 런타임/SDK 설치 방법
13165정성태11/18/20224880개발 환경 구성: 650. Azure - "scm" 프로세스와 엮인 서비스 모음
13164정성태11/18/20225825개발 환경 구성: 649. Azure - 비주얼 스튜디오를 이용한 AppService 원격 디버그 방법
13163정성태11/17/20225730개발 환경 구성: 648. 비주얼 스튜디오에서 안드로이드 기기 인식하는 방법
13162정성태11/15/20226779.NET Framework: 2069. .NET 7 - AOT(ahead-of-time) 컴파일
13161정성태11/14/20226034.NET Framework: 2068. C# - PublishSingleFile로 배포한 이미지의 역어셈블 가능 여부 (난독화 필요성) [4]
13160정성태11/11/20225956.NET Framework: 2067. C# - PublishSingleFile 적용 시 native/managed 모듈 통합 옵션
13159정성태11/10/20229272.NET Framework: 2066. C# - PublishSingleFile과 관련된 옵션 [3]
13158정성태11/9/20225436오류 유형: 826. Workload definition 'wasm-tools' in manifest 'microsoft.net.workload.mono.toolchain' [...] conflicts with manifest 'microsoft.net.workload.mono.toolchain.net7'
13157정성태11/8/20226079.NET Framework: 2065. C# - Mutex의 비동기 버전파일 다운로드1
13156정성태11/7/20227014.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점파일 다운로드1
13155정성태11/4/20226474디버깅 기술: 183. TCP 동시 접속 (연결이 아닌) 시도를 1개로 제한한 서버
13154정성태11/3/20225955.NET Framework: 2063. .NET 5+부터 지원되는 GC.GetGCMemoryInfo파일 다운로드1
13153정성태11/2/20227259.NET Framework: 2062. C# - 코드로 재현하는 소켓 상태(SYN_SENT, SYN_RECV)
13152정성태11/1/20225844.NET Framework: 2061. ASP.NET Core - DI로 추가한 클래스의 초기화 방법 [1]
13151정성태10/31/20226023C/C++: 161. Windows 11 환경에서 raw socket 테스트하는 방법파일 다운로드1
13150정성태10/30/20225989C/C++: 160. Visual Studio 2022로 빌드한 C++ 프로그램을 위한 다른 PC에서 실행하는 방법
13149정성태10/27/20225954오류 유형: 825. C# - CLR ETW 이벤트 수신이 GCHeapStats_V1/V2에 대해 안 되는 문제파일 다운로드1
13148정성태10/26/20225899오류 유형: 824. msbuild 에러 - error NETSDK1005: Assets file '...\project.assets.json' doesn't have a target for 'net5.0'. Ensure that restore has run and that you have included 'net5.0' in the TargetFramew
13147정성태10/25/20224972오류 유형: 823. Visual Studio 2022 - Unable to attach to CoreCLR. The debugger's protocol is incompatible with the debuggee.
13146정성태10/24/20225837.NET Framework: 2060. C# - Java의 Xmx와 유사한 힙 메모리 최댓값 제어 옵션 HeapHardLimit
... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...