Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 8개 있습니다.)
.NET Framework: 268. .NET Array는 왜 12bytes의 기본 메모리를 점유할까?
; https://www.sysnet.pe.kr/2/0/1173

.NET Framework: 269. 일반 참조형의 기본 메모리 소비는 얼마나 될까요?
; https://www.sysnet.pe.kr/2/0/1174

.NET Framework: 270. .NET 참조 개체 인스턴스의 Object Header를 확인하는 방법
; https://www.sysnet.pe.kr/2/0/1175

.NET Framework: 271. C#에서 확인해 보는 관리 힙의 인스턴스 구조
; https://www.sysnet.pe.kr/2/0/1176

.NET Framework: 401. windbg에서 확인해 보는 관리 힙의 인스턴스 구조
; https://www.sysnet.pe.kr/2/0/1559

.NET Framework: 1003. x64 환경에서 참조형의 기본 메모리 소비는 얼마나 될까요?
; https://www.sysnet.pe.kr/2/0/12486

.NET Framework: 1004. C# - GC Heap에 위치한 참조 개체의 주소를 알아내는 방법
; https://www.sysnet.pe.kr/2/0/12487

.NET Framework: 1184. C# - GC Heap에 위치한 참조 개체의 주소를 알아내는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/13017




C# - GC Heap에 위치한 참조 개체의 주소를 알아내는 방법

예전에 아래의 글을 설명하면서,

C#에서 확인해 보는 관리 힙의 인스턴스 구조
; https://www.sysnet.pe.kr/2/0/1176

참조 개체가 관리 힙의 어디에 위치하고 있는지 그 주솟값을 구하기 위해 fixed를 사용하려고 억지로 필드 하나를 정의하는 코드를 사용했는데요,

// x86 환경
class Program
{
    public int value = 0xff;

    static unsafe void Main(string[] args)
    {
        Program pg = new Program();

        fixed (int* p2 = &pg.value)
        {
            Console.WriteLine("OBJECT Header(SyncBlock Index): " + (*(p2 - 2)).ToString("x"));
            Console.WriteLine("MethodTable Address: " + (*(p2 - 1)).ToString("x"));
            Console.WriteLine("첫 번째 필드 값: " + (*(p2 - 0)).ToString("x"));
        }
    }
}

이후로 몇몇 글에서 __makeref를 이용해 참조 주솟값을 구하는 방법을 사용했었지만,

windbg/sos - Dictionary의 entries 배열 내용을 모두 덤프하는 방법 (do_hashtable.py)
; https://www.sysnet.pe.kr/2/0/12087

직접적으로 명시를 하지 않아 검색이 잘 되는군요. ^^ 그래서 검색의 목적을 위해, 아울러 혹시 궁금하신 분들을 위해 별도 제목으로 뽑아 이렇게 기록을 남깁니다. 방법은 간단하게 TypedReference와 __makeref를 이용해 다음과 같이 구할 수 있습니다.

// x86 or x64
using System;
using System.Threading;

namespace ConsoleApplication1
{
    class Program
    {
        public int value = 0xff;

        static unsafe void Main(string[] args)
        {
            Program pg = new Program();

            IntPtr ptr = GetRefAddress(pg);
        }

        private unsafe static IntPtr GetRefAddress(object obj)
        {
            TypedReference refA = __makeref(obj);
            return **(IntPtr**)&refA;
        }

        /* 또는, Dumping the managed heap in C#
        public static nint GetAddress(T obj)
        {
            // Get the address of the reference, cast it to a pointer,
            // then dereference it to get the address of the object
            return (nint)(*(T**)&obj);
        }
        */
    }
}

이 코드를 기반으로 "C#에서 확인해 보는 관리 힙의 인스턴스 구조" 글에 사용한 코드를 다시 작성해 보면,

static unsafe void WriteRefObjectInfo(IntPtr pAddress)
{
    IntPtr pHeader = pAddress - (IntPtr.Size * 1);
    Console.WriteLine("OBJECT Header(SyncBlock Index): " + GetWordAsText(pHeader));

    Console.WriteLine("MethodTable Address: " + GetWordAsText(pAddress));

    IntPtr pIntValue = pAddress + (IntPtr.Size * 1);
    Console.WriteLine("첫 번째 int 필드 값: " + GetWordAsText(pIntValue));
}

static unsafe string GetWordAsText(IntPtr pAddress)
{
    if (IntPtr.Size == 4)
    {
        return (*((int*)pAddress.ToPointer())).ToString("x");
    }
    else
    {
        return (*((long*)pAddress.ToPointer())).ToString("x");
    }
}

다음과 같이 값이 잘 출력되는 것을 확인할 수 있습니다.

OBJECT Header(SyncBlock Index): 0
MethodTable Address: 7ff8566f5a70
첫 번째 int 필드 값: ff

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






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

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

비밀번호

댓글 작성자
 




... 166  167  168  169  170  171  [172]  173  174  175  176  177  178  179  180  ...
NoWriterDateCnt.TitleFile(s)
834정성태1/25/201031354.NET Framework: 172. WCF - webHttpBinding 윈도우 인증 구현 예제 [3]파일 다운로드1
833정성태1/25/201030671.NET Framework: 171. WCF - webHttpBinding 구현 예제 [1]파일 다운로드1
832정성태1/25/201033978.NET Framework: 170. PerformanceCounter의 RawValue/NextValue()에서 멈춤 현상
831정성태1/14/201023796개발 환경 구성: 70. WSS - check out 메뉴에서 오류나는 문제
830정성태1/10/201028909개발 환경 구성: 69. Windows Internal Database
829정성태1/7/201028245개발 환경 구성: 68. ODP.NET + OraMTS 사용
828정성태1/7/201035360개발 환경 구성: 67. 환경 변수를 이용한 다중 ODAC 버전 테스트
827정성태1/4/201028183개발 환경 구성: 66. .NET 응용 프로그램에서 64비트 Oracle Data Access Components 사용 (2)
826정성태1/3/201035700기타: 28. 2009년 인기 순위 정리
825정성태1/3/201052590개발 환경 구성: 65. .NET 응용 프로그램에서 64비트 Oracle Data Access Components 사용 [5]
823정성태1/1/201029515개발 환경 구성: 64. ODP.NET 설치 작업 없이 ASP.NET 응용 프로그램 배포파일 다운로드1
822정성태1/1/201093484개발 환경 구성: 63. ODP.NET 설치 없이 .NET 클라이언트 프로그램 배포하는 방법 [6]파일 다운로드2
821정성태1/1/201054591개발 환경 구성: 62. .NET 응용 프로그램에서 Oracle XE 사용 [1]
820정성태12/29/200926495개발 환경 구성: 61. Oxite 소스 코드를 Visual Studio 2010으로 마이그레이션파일 다운로드1
818정성태12/27/200924830개발 환경 구성: 60. Cassini 서버를 localhost 이외의 주소에서 접근하도록 변경
817정성태12/21/200926109개발 환경 구성: 59. WebDev 2.0에서 실행하는 Pet Shop 4.0
816정성태12/19/200927912개발 환경 구성: 58. Pet Shop 4.0을 IIS 없이 실행하는 방법 [1]파일 다운로드1
815정성태12/18/200930605개발 환경 구성: 57. Pet Shop 4.0 - SQL Server Compact Edition Version - 두 번째 이야기파일 다운로드1
814정성태12/16/200926368오류 유형: 91. VS2010 beta2 - The application cannot start
812정성태12/11/200930575개발 환경 구성: 56. Pet Shop 4.0 - SQL Server Compact Edition Version파일 다운로드2
811정성태12/2/200945109.NET Framework: 169. [in, out] 배열을 C#에서 C/C++로 넘기는 방법 - 두 번째 이야기 [8]파일 다운로드2
810정성태12/1/200940220.NET Framework: 168. [in,out] 배열을 C#에서 C/C++로 넘기는 방법 [3]
809정성태11/27/200929478오류 유형: 90. Method not found: 'Void System.Reflection.Emit.DynamicMethod..ctor(System.String, System.Type, System.Type[])'.
808정성태11/26/200938364VC++: 38. X64 빌드 오류: error LNK2001: unresolved external symbol [COMDLL]_ProxyFileInfo
807정성태11/23/200932081웹: 14. 로컬에 있는 HTML 페이지를 보호 모드에서 여는 방법
802정성태11/19/200934304.NET Framework: 167. 다른 스레드의 호출 스택 덤프 구하는 방법파일 다운로드1
... 166  167  168  169  170  171  [172]  173  174  175  176  177  178  179  180  ...