I am using SOS debug extension dll to check the memory layout of a String type, and below is the result.
!dso
ESP/REG Object Name
0015EFC0 01c6b9cc System.String hello,world
!do 01c6b9cc
Name: System.String
MethodTable: 6de3f9ac
EEClass: 6db78bb0
Size: 36(0x24) bytes
File: C:\Windows\Microsoft.Net\assembly\GAC_32\mscorlib\v4.0_4.0.0.0__b77a5c561934e089>\mscorlib.dll
String: hello,world
Fields:
MT Field Offset Type VT Attr Value Name
6de42978 40000ed 4 System.Int32 1 instance 11 m_stringLength
6de41dc8 40000ee 8 System.Char 1 instance 68 m_firstChar
6de3f9ac 40000ef 8 System.String 0 shared static Empty
>> Domain:Value 00331488:01c61228 <<
Now I am wondering, where exactly is the string value "hello world" stored?
Thanks.
At m_firstChar. The heap allocation is large enough to fit the entire string, not just the first character. Easy to see in Visual Studio as well:
When the breakpoint hits, use Debug + Windows + Memory + Memory1. In the Address box type s. You'll see:
This is for .NET 3.5 SP1. You won't see the m_arrayLength member in .NET 4.0 and up, the field was removed.
Like a C "string", it's stored in the
m_stringLength
bytes starting atm_firstChar
which is an unsafe pointer, not an actual character. C# uses a length prefixed string rather than anull
delimited one though.That said, the beauty of the CLR is that you don't need to care. How has this become an issue?