- Frugal Cafe
- Posts
- FC25: ICSharpCode.ILSpy.CSharpLanguage
FC25: ICSharpCode.ILSpy.CSharpLanguage
Magic of StringList replacing StringBuilder
data:image/s3,"s3://crabby-images/a56fe/a56feb345f6efa697e9c754aef489c362b027baf" alt=""
I’m an active user of ILSpy to understand .Net implementations and search for their usages. So ILSpy is always open on my machine. Noticed it’s now running in .Net Core, I captured a trace and found this allocation stat:
data:image/s3,"s3://crabby-images/bfc0d/bfc0d1ee0c91ec7e5f82ea18fafec30ca1d6f697" alt=""
StringBuilder accounts for 10% of total allocations, so there are StringBuilder usage issues. Here is the stack found:
data:image/s3,"s3://crabby-images/33976/33976c1fc5c84c6b1d0f3741e21b153aa97dc834" alt=""
CSharpLanguage.ToCSharpString is responsible for 33.5% of total allocations. Notice here we’ve an unusual call: StringBuilder.Insert. Source code:
data:image/s3,"s3://crabby-images/fc272/fc272f4b56ca0e07d4bdc17b788cf1f5ad13529d" alt=""
This is a very unique pattern of using StringBuilder: all the calls are to StringBuilder.Insert(0, x). Let’s replace StringBuilder with the StringList class we created before (FC20: Need a List replacement? (beehiiv.com)):
data:image/s3,"s3://crabby-images/4fb8f/4fb8f61180c65b24b7e8266dde241ce38bc2938f" alt=""
StringBuilder.Insert(0, x) is now replaced by StringList.Add, StringBuilder.ToString is replaced with StringList.ReverseConcat. Implementation:
data:image/s3,"s3://crabby-images/cc2ff/cc2ff2fe61e0a9f87167738590a1b7ca5ffa4de5" alt=""
In best case where just a single name is needed without namespace, there is no not even a single allocation. The StringList is being reused here.
We should be able to remove most allocations here. Source code here: https://github.com/ProfFrugal/FrugalCafe/blob/master/Posts/CSharpLanguage.cs