Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

XNA Content 리소스의 해제 후 다시 로드해서 사용하면 ObjectDisposedException 예외 발생

아래와 같은 질문이 있군요. ^^

c# 으로 게임을 제작중인데요 리소스 파일을 동적으로 관리하고싶은데 문제점이.; 
; http://www.gamecodi.com/board/zboard.php?id=GAMECODI_Talkdev&no=2672&z=9

재현 방법은 간단합니다. 리소스를 하나 로드한 다음, Dispose하고 다시 로드하면 여지없이 동일한 오류가 발생합니다. 다음은 예전에 실습했던 Shader 예제에서 Effect 리소스로 테스트한 것입니다.

protected override void LoadContent()
{
    // ... 생략 ...
    _colorShader = Content.Load<Effect>("ColorShader");
    _colorShader.Dispose();

    _colorShader = Content.Load<Effect>("ColorShader");
}

이렇게 처리하고 실행하면 Draw 단계에서 예외가 발생합니다.

이에 대해 검색해 보면, XNA 내부적으로 리소스를 캐시하기 때문에 Dispose를 했다고 해서 cache 정보까지 삭제되는 것은 아니라고 합니다. 그런데, 하나같이 그에 대한 해결책이 없군요. ^^;

자... 그래서 오랜만에 .NET Reflector를 이용해서 소스코드가 없는 XNA 어셈블리의 코드를 디버깅으로 들어가 보았습니다.

.NET Reflector 를 이용한 "소스 코드가 없는" 어셈블리 디버깅
; https://www.sysnet.pe.kr/2/0/1201

다음은 Load 코드인데,

public virtual T Load<T>(string assetName)
{
    object obj2;
    Logger.BeginLogEvent(LoggingEvent.LoadContent, "XNA: Begin Loading Content: " + assetName);
    if (this.loadedAssets == null)
    {
        throw new ObjectDisposedException(this.ToString());
    }
    if (string.IsNullOrEmpty(assetName))
    {
        throw new ArgumentNullException("assetName");
    }
    assetName = TitleContainer.GetCleanPath(assetName);
    if (this.loadedAssets.TryGetValue(assetName, out obj2))
    {
        if (!(obj2 is T))
        {
            throw new ContentLoadException(string.Format(CultureInfo.CurrentCulture, FrameworkResources.BadXnbWrongType, new object[] { assetName, obj2.GetType(), typeof(T) }));
        }
        return (T) obj2;
    }
    T local = this.ReadAsset<T>(assetName, null);
    this.loadedAssets.Add(assetName, local);
    Logger.EndLogEvent(LoggingEvent.LoadContent, "XNA: Done Loading Content: " + assetName);
    return local;
}

함수 내부에서 Step-trace를 하며 디버깅을 진행해 보면, 내부적인 cache 역할을 loadedAssets 변수를 통해 하는 것을 확인할 수 있습니다. 새로운 리소스 이름으로 Content.Load를 호출하면 TryGetValue에서 false가 나오므로 this.ReadAsset 코드로 넘어가지만, 해당 리소스를 Dispose 했다고 해서 this.loadedAssets 컬렉션에는 여전히 리소스 이름이 남아 있기 때문에 그 이후의 Load 호출에서는 새롭게 리소스를 로드하지 않고 실패하게 되는 것입니다.

원인을 알았으니, 그럼 this.loadedAssets에서 리소스 이름을 삭제하는 코드가 있는지 살펴보면 될텐데요. 아쉽게도 ... 없습니다. ^^; 네, 없습니다. 믿으세요. 없습니다. 그래서 다들 해결책을 얘기하지 못했던 것입니다.

이상하군요. 충분히 요구 사항이 있었을텐데... 어쨌든, 없으므로 이제 남은 방법은 Reflection의 힘을 빌리는 것! ^^

그래서, 다음과 같이 loadedAssets에서 리소스 이름으로 캐시를 날리는 작업을 해줍니다.

private void RemoveAsset(string assetName)
{
    Type type = this.Content.GetType();

    FieldInfo fieldInfo = type.GetField("loadedAssets", BindingFlags.Instance | BindingFlags.NonPublic);
    Dictionary<string, object> loadedAssets =
        fieldInfo.GetValue(this.Content) as Dictionary<string, object>;

    if (loadedAssets != null)
    {
        loadedAssets.Remove(assetName);
    }
}

그다음, 요렇게 코드를 수정해 주면 끝!

protected override void LoadContent()
{
    // ... 생략 ...
    _colorShader = Content.Load<Effect>("ColorShader");
    _colorShader.Dispose();

    RemoveAsset("ColorShader");

    _colorShader = Content.Load<Effect>("ColorShader");
}




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







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

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

비밀번호

댓글 작성자
 



2014-05-16 05시22분
[signal] 안녕하세요 ^^ 정말 많은 도움이되었습니다 ㅋ 테스트 해본결과 잘되구요 ㅎ
그런데 테스트 하다가 궁금한점이 생겨서요 . 저기 만약에 생성을 폴더생성을해서 그안에서 했을경우
(test 라는 폴더 안의경우)
protected override void LoadContent()
{
    // ... 생략 ...
    _colorShader = Content.Load<Effect>("test/ColorShader");
    _colorShader.Dispose();

    RemoveAsset("test/ColorShader");

    _colorShader = Content.Load<Effect>("test/ColorShader");
}

이렇게 경로가 들어가게될텐데요 이경우에는 에러가 나는데 이이유는 무엇인가요 ?
경로를 다 넣어줘도 마지막 파일명만 써도 . 폴더 안에 있을경우에는 에러가 납니다 .
어떤이유에서 나는건지 알고셰신지요? ^^
[guest]
2014-05-16 01시46분
신기해서 방금 확인을 해봤습니다. ^^ 보니까, TitleContainer.GetCleanPath 메서드가 assetName의 '/' 문자를 디렉토리 구분자(\\)로 치환하는 역할을 합니다. 따라서, 다음과 같이 바꿔주시면 됩니다.

            string resName = "Test\\ColorShader";
            _colorShader = Content.Load<Effect>(resName);
            _colorShader.Dispose();
            RemoveAsset(resName);

            _colorShader = Content.Load<Effect>(resName);
정성태

... 91  92  93  94  95  96  97  98  99  [100]  101  102  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11494정성태4/16/201816928개발 환경 구성: 362. Azure Web Apps(App Services)에 사용자 DNS를 지정하는 방법
11493정성태4/16/201818671개발 환경 구성: 361. Azure Web App(App Service)의 HTTP/2 프로토콜 지원
11492정성태4/13/201816802개발 환경 구성: 360. Azure Active Directory의 사용자 도메인 지정 방법
11491정성태4/13/201820152개발 환경 구성: 359. Azure 가상 머신에 Web Application을 배포하는 방법
11490정성태4/12/201818841.NET Framework: 739. .NET Framework 4.7.1의 새 기능 - Configuration builders [1]파일 다운로드1
11489정성태4/12/201816613오류 유형: 463. 윈도우 백업 오류 - a Volume Shadow Copy Service operation failed.
11488정성태4/12/201820980오류 유형: 462. Unhandled Exception in Managed Code Snap-in - FX:{811FD892-5EB4-4E73-A147-F1E079E36C4E}
11487정성태4/12/201818495디버깅 기술: 115. windbg - 닷넷 메모리 덤프에서 정적(static) 필드 값을 조사하는 방법
11486정성태4/11/201817581오류 유형: 461. Error MSB4064 The "ComputeOutputOnly" parameter is not supported by the "VsTsc" task
11485정성태4/11/201826257.NET Framework: 738. C# - Console 프로그램이 Ctrl+C 종료 시점을 감지하는 방법파일 다운로드1
11484정성태4/11/201827891.NET Framework: 737. C# - async를 Task 타입이 아닌 사용자 정의 타입에 적용하는 방법파일 다운로드1
11483정성태4/10/201831000개발 환경 구성: 358. "Let's Encrypt"에서 제공하는 무료 SSL 인증서를 IIS에 적용하는 방법 (2) [1]
11482정성태4/10/201822465VC++: 126. CUDA Core 수를 알아내는 방법
11481정성태4/10/201835386개발 환경 구성: 357. CUDA의 인덱싱 관련 용어 - blockIdx, threadIdx, blockDim, gridDim
11480정성태4/9/201825144.NET Framework: 736. C# - API를 사용해 Azure에 접근하는 방법 [2]파일 다운로드1
11479정성태4/9/201819510.NET Framework: 735. Azure - PowerShell로 Access control(IAM)에 새로운 계정 만드는 방법
11478정성태11/8/201922932디버깅 기술: 115. windbg - 덤프 파일로부터 PID와 환경변수 등의 정보를 구하는 방법 [1]
11477정성태4/8/201819694오류 유형: 460. windbg - sos 명령어 수행 시 c0000006 오류 발생
11476정성태4/8/201821040디버깅 기술: 114. windbg - !threads 출력 결과로부터 닷넷 관리 스레드(System.Threading.Thread) 객체를 구하는 방법
11475정성태3/28/201824118디버깅 기술: 113. windbg - Thread.Suspend 호출 시 응용 프로그램 hang 현상에 대한 덤프 분석
11474정성태3/27/201822589오류 유형: 459. xperf: error: TEST.Event: Invalid flags. (0x3ec).
11473정성태3/22/201827400.NET Framework: 734. C# - Thread.Suspend 호출 시 응용 프로그램 hang 현상파일 다운로드2
11472정성태3/22/201820529개발 환경 구성: 356. GTX 1070, GTX 960, GT 640M의 cudaGetDeviceProperties 출력 결과
11471정성태3/20/201823462VC++: 125. CUDA로 작성한 RGB2RGBA 성능 [1]파일 다운로드1
11470정성태3/20/201826730오류 유형: 458. Visual Studio - CUDA 프로젝트 빌드 시 오류 C1189, expression must have a constant value
11469정성태3/19/201819823오류 유형: 457. error MSB3103: Invalid Resx file. Could not load file or assembly 'System.Windows.Forms, ...' or one of its dependencies.
... 91  92  93  94  95  96  97  98  99  [100]  101  102  103  104  105  ...