Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)

Entity Framework 4.1 - CodeFirst 개체의 직렬화 시 순환 참조 해결하는 방법

우선, 순환 참조가 직렬화 시에 어떻게 발생할 수 있는지 다음의 글을 한번 읽어보시고 시작하는 것도 좋겠습니다. ^^

순환참조와 XmlSerializer
; https://www.sysnet.pe.kr/2/0/751

위의 XmlSerializer에서는 순환참조가 발생하는 경우 System.StackOverflowException 예외가 발생하지만 WCF의 경우에는 System.Runtime.Serialization.SerializationException 수준에서 끝납니다. 아마도 내부적인 threshold 값의 제약으로 스레드의 스택이 바닥나기 전에 중지되기 때문인 것 같습니다. (정확히 어떤 threshold 값에 해당하는지 아직 잘 모르겠습니다. ^^)

본론으로 들어가서, 이번에는 Entity Framework의 CodeFirst에서 순환참조가 발생하는 경우를 살펴볼 텐데요. 실제로 예제를 만들어서 재현을 하고 문제를 해결하는 식으로 진행해 보겠습니다.

우선, 예제 코드는 지난번 글에서 만들어 둔 EF + WCF 프로젝트로 만들어 두었던 것을 재사용할 텐데요, 아래의 글에서 다운로드하시면 됩니다.

Entity Framework 4.1 - Code First + WCF 서비스 시 EndpointNotFoundException 오류 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1085

codefirst_example.zip
; https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=613&boardid=331301885

이제, MyEntity의 목록을 반환하는 WCF 메서드를 한번 볼까요?

public MyEntity [] GetMyEntities()
{
    using (DBContext db = new DBContext())
    {
        var items = from record in db.MyEntities
                    select record;

        var list = items.ToArray();
        return list;
    }
}

짐작하시겠지만, 이렇게 반환된 MyEntity는 SubEntities 속성이 null로 되어 있습니다. 만약, 그 속성값을 채우고 싶다면 Linq 쿼리를 다음과 같이 변경해 주어야 합니다.

public MyEntity [] GetMyEntities()
{
    using (DBContext db = new DBContext())
    {
        var items = from record in db.MyEntities.Include("SubEntities")
                    select record;

        var list = items.ToArray();
        return list;
    }
}

자, 이렇게 코드를 변경하고 WCF 서비스를 사용하면 여지없이 클라이언트 측에서 다음과 같은 예외를 받게 됩니다.

Unhandled Exception: System.ServiceModel.CommunicationException: An error occurred while receiving the HTTP response to http://.../Service.svc. This could be due to the service endpoint binding not using the HTTP protocol. This could also be due to an HTTP request context being aborted by the server (possibly due to the service shutting down). See server logs for more details. ---> System.Net.WebException: The underlying connection was closed: An unexpected error occurred on a receive. ---> System.IO.IOException: Unable to read data from the transport connection: An existing connection was forcibly closed by the remote host. ---> System.Net.Sockets.SocketException: An existing connect ion was forcibly closed by the remote host
at System.Net.Sockets.NetworkStream.Read(Byte[] buffer, Int32 offset, Int32 size)
--- End of inner exception stack trace ---
at System.Net.Sockets.NetworkStream.Read(Byte[] buffer, Int32 offset, Int32 size)
at System.Net.PooledStream.Read(Byte[] buffer, Int32 offset, Int32 size)
at System.Net.Connection.SyncRead(HttpWebRequest request, Boolean userRetrievedStream, Boolean probeRead)
--- End of inner exception stack trace ---
at System.Net.HttpWebRequest.GetResponse()
at System.ServiceModel.Channels.HttpChannelFactory.HttpRequestChannel.HttpChannelRequest.WaitForReply(TimeSpan timeout)
--- End of inner exception stack trace ---


보는 바와 같이, 클라이언트 측에서의 오류 메시지로는 저번에 다룬 DynamicProxies 직렬화 오류와 비교해서 차이가 없습니다. 다시 한번 이 오류의 정확한 원인을 알기 위해서 DataContractSerializer를 이용하여 직접 직렬화 시도를 하고 예외 메시지를 출력하도록 합니다.

public MyEntity [] GetMyEntities()
{
    ...[생략]...
    try
    {
        MemoryStream ms = new MemoryStream();
        DataContractSerializer dcs = new DataContractSerializer(typeof(MyEntity[]));
        dcs.WriteObject(ms, list);
    }
    catch (Exception ex)
    {
        System.Diagnostics.Debug.WriteLine(ex.ToString());
    }

    return list;
}

실행해 보면, 아래와 같이 순환 참조에 걸렸음을 알려주는 오류 메시지가 발견됩니다.

System.Runtime.Serialization.SerializationException: Object graph for type 'System.Collections.Generic.HashSet`1[[SubEntity, App_Code.xdcj8jsd, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null]]' contains cycles and cannot be serialized if reference tracking is disabled. 
    at System.Runtime.Serialization.XmlObjectSerializerWriteContext.OnHandleReference(XmlWriterDelegator xmlWriter, Object obj, Boolean canContainCyclicReference) 
    at System.Runtime.Serialization.XmlObjectSerializerWriteContext.InternalSerializeReference(XmlWriterDelegator xmlWriter, Object obj, Boolean isDeclaredType, Boolean writeXsiType, Int32 declaredTypeID, RuntimeTypeHandle declaredTypeHandle) 
    at WriteEventSourceToXml(XmlWriterDelegator , Object , XmlObjectSerializerWriteContext , ClassDataContract ) 
    at System.Runtime.Serialization.ClassDataContract.WriteXmlValue(XmlWriterDelegator xmlWriter, Object obj, XmlObjectSerializerWriteContext context) 
    ...[이하, 순환 참조로 인한 한참 동안의 예외 메시지 콜 스택 출력]...

이에 대한 원인은 MyEntity, SubEntity 정의를 보면 쉽게 찾을 수 있습니다.

public class MyEntity
{
    [DatabaseGeneratedAttribute(DatabaseGeneratedOption.Identity)]
    public int Id { get; set; }
    public string Description { get; set; }

    public ICollection<SubEntity> SubEntities { get; set; }
}

public class SubEntity
{
    [DatabaseGeneratedAttribute(DatabaseGeneratedOption.Identity)]
    public int Id { get; set; }
    public string Description { get; set; }

    public int MyEntityId { get; set; }
    public MyEntity MyEntity { get; set; }
}

전형적인 순환 참조의 예인데요. 하지만, Entity Framework이 데이터베이스에 대한 OR-Mapping 도구임을 감안하면 위의 코드처럼 나와야 하는 것이 너무나 당연한 것도 사실입니다. 단지, 이렇게 정의된 DbContext가 지난번 DbContext.Configuration.ProxyCreationEnabled 속성 관련한 문제에서와 마찬가지로 WCF 서비스로 내보내기 하는 용도로는 적합하지 않다는 점입니다.

그럼 어떻게 해결해야 할까요? 2가지 방법이 있는데, 첫 번째로는 다음과 같이 코드를 변경함으로써 쉽게 해결이 가능합니다.

public MyEntity [] GetMyEntities()
{
    using (DBContext db = new DBContext())
    {
        var items = from record in db.MyEntities.Include("SubEntities")
                    select record;

        foreach (var item in items)
        {
            foreach (var subEntity in item.SubEntities)
            {
                subEntity.MyEntity = null;
            }
        }

        return items.ToArray();
    }
}

음... 일단은 너무나 쉬운 방법이라서 적용하기에 좋겠지만, 왠지 프로그래머의 직감으로 볼 때 뭔가 꺼림직한 면이 있습니다. 이를 해결할 수 있는 두 번째 방법은... ^^ 아쉽지만 지면(?) 관계상 다음에 살펴보도록 하겠습니다.

첨부된 파일은 여기까지 변경된 코드를 포함하고 있습니다.




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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2011-07-15 07시03분
Entity Framework 4.1 - CodeFirst 개체의 직렬화 시 순환 참조 해결하는 방법 - 두 번째 이야기
; http://www.sysnet.pe.kr/2/0/1087
정성태

... 151  [152]  153  154  155  156  157  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1250정성태2/27/201231331VC++: 59. C/C++ 프로젝트 빌드 속도 개선 - UnityBuild를 아세요? [3]
1249정성태2/26/201230996.NET Framework: 311. .NET 스레드 콜 스택 덤프 (5) - ICorDebug 인터페이스 사용법 [2]파일 다운로드3
1248정성태2/25/201242486.NET Framework: 310. C#의 Shift 비트 연산 정리파일 다운로드1
1247정성태2/25/201225217.NET Framework: 309. .NET 응용 프로그램에 기본 생성되는 스레드들에 대한 탐구 [1]파일 다운로드1
1246정성태2/25/201224775개발 환경 구성: 145. 한영 변환은 되지만, 정작 한글 입력이 안되는 경우
1245정성태2/25/201235478개발 환경 구성: 144. 윈도우에서도 유닉스처럼 명령행으로 원격 접속하는 방법
1244정성태2/24/201232656.NET Framework: 308. .NET System.Threading.Thread 개체에서 Native Thread Id를 구할 수 있을까? [1]파일 다운로드1
1243정성태2/23/201232608개발 환경 구성: 143. Visual Studio 2010 - .NET Framework 소스 코드 디버깅 - 두 번째 이야기 [1]
1242정성태2/20/201239480VC++: 58. API Hooking - 64비트를 고려해야 한다면? EasyHook! [7]파일 다운로드1
1241정성태2/20/201226322.NET Framework: 307. .NET 4.0 응용 프로그램을 위한 ILMerge
1240정성태2/19/201232574디버깅 기술: 48. C/C++ JNI DLL을 Visual Studio로 디버깅하는 방법 [2]
1239정성태2/19/201224225.NET Framework: 306. 컴퓨터에 실행된 프로세스 중에 닷넷 응용 프로그램임을 알 수 있는 방법 - C# [1]파일 다운로드1
1238정성태2/19/201228104.NET Framework: 305. GetPrivateProfileSection / WritePrivateProfileSection의 C# 버전파일 다운로드1
1237정성태2/18/201232408개발 환경 구성: 142. Windows Embedded POSReady 7 설치 [1]
1236정성태2/17/201228239개발 환경 구성: 141. Windows 2008 R2 RDP 라이선스 서버 설치하는 방법
1235정성태2/16/201226686.NET Framework: 304. Hyper-V의 가상 머신을 C#으로 제어하는 방법 [1]파일 다운로드1
1234정성태2/16/201227109.NET Framework: 303. 원본 파일의 공백/라인을 유지한 체 XML 파일을 저장하는 방법 [1]파일 다운로드1
1233정성태2/16/201233192.NET Framework: 302. supportedRuntime 옵션과 System.BadImageFormatException 예외 [5]
1232정성태2/9/201229073VC++: 57. 웹 브라우저에서 Flash만 빼고 다른 ActiveX를 차단할 수 있을까? [3]파일 다운로드1
1231정성태2/8/201238629VC++: 56. Win32 API 후킹 - Trampoline API Hooking [5]파일 다운로드1
1230정성태2/6/201223975개발 환경 구성: 140. 프로젝트 생성 시부터 "Enable the Visual Studio hosting process" 옵션을 끄는 방법
1229정성태2/4/201229001.NET Framework: 301. P/Invoke의 성능을 높이기 위해 C++/CLI가 선택되려면? [5]파일 다운로드1
1228정성태2/4/201278320.NET Framework: 300. C#으로 만드는 음성인식/TTS 프로그램 [47]파일 다운로드1
1227정성태2/3/201229205.NET Framework: 299. 해당 어셈블리가 Debug 빌드인지, Release 빌드인지 알아내는 방법파일 다운로드1
1226정성태1/28/201270099.NET Framework: 298. 홀 펀칭(Hole Punching)을 이용한 Private IP 간 통신 - C# [15]파일 다운로드3
1225정성태1/24/201225694.NET Framework: 297. 특정 EXE 파일의 실행을 Internet Explorer처럼 "Protected Mode"로 실행하는 방법 [1]파일 다운로드1
... 151  [152]  153  154  155  156  157  158  159  160  161  162  163  164  165  ...