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.
Both the string and the char[] are stored on the heap - so storage is the same. Internally I would assume a string simply is a cover for char[] with lots of extra code to make it useful for you.
In C#, IndexOf() method is a string method. This method is used to find the zero-based index of the first occurrence of a specified character or string within the current instance of the string. The method returns -1 if the character or string is not found.
A string is an object of type String whose value is text. Internally, the text is stored as a sequential read-only collection of Char objects. There's no null-terminating character at the end of a C# string; therefore a C# string can contain any number of embedded null characters ('\0').
You can't append to a string, and you can't add a specific character to a string. Strings cannot be modified. You can create a new string based on an existing string.
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:
class Program {
static void Main(string[] args) {
string s = "hello" + "world";
} // <=== Breakpoint here
}
When the breakpoint hits, use Debug + Windows + Memory + Memory1. In the Address box type s. You'll see:
0x01B3F6BC e8 0a 67 6e 0b 00 00 00 0a 00 00 00 68 00 65 00 è.gn........h.e.
0x01B3F6CC 6c 00 6c 00 6f 00 77 00 6f 00 72 00 6c 00 64 00 l.l.o.w.o.r.l.d.
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 at m_firstChar
which is an unsafe pointer, not an actual character. C# uses a length prefixed string rather than a null
delimited one though.
That said, the beauty of the CLR is that you don't need to care. How has this become an issue?
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With