String.Format
内部で使用StringBuilder
されるため、論理的には、オーバーヘッドが増えるためにパフォーマンスが少し低下するという考えにつながります。ただし、単純な文字列連結は、1 つの文字列を他の 2 つの文字列の間に挿入する最速の方法です。この証拠は、何年も前にリコ マリアーニが最初のパフォーマンス クイズで示しました。単純な事実は、文字列部分の数がわかっている場合 (制限なし — 常に 1000 部分であることがわかっている限り、1000 部分を連結できます)、連結は常にStringBuilder
orよりも高速ですString.Format
。これらは、単一のメモリ割り当てと一連のメモリ コピーで実行できます。これが証拠です。
String.Concat
最終的に を呼び出すいくつかのメソッドの実際のコードは次のFillStringChecked
とおりです。これはポインタを使用してメモリをコピーします (Reflector を介して抽出)。
public static string Concat(params string[] values)
{
int totalLength = 0;
if (values == null)
{
throw new ArgumentNullException("values");
}
string[] strArray = new string[values.Length];
for (int i = 0; i < values.Length; i++)
{
string str = values[i];
strArray[i] = (str == null) ? Empty : str;
totalLength += strArray[i].Length;
if (totalLength < 0)
{
throw new OutOfMemoryException();
}
}
return ConcatArray(strArray, totalLength);
}
public static string Concat(string str0, string str1, string str2, string str3)
{
if (((str0 == null) && (str1 == null)) && ((str2 == null) && (str3 == null)))
{
return Empty;
}
if (str0 == null)
{
str0 = Empty;
}
if (str1 == null)
{
str1 = Empty;
}
if (str2 == null)
{
str2 = Empty;
}
if (str3 == null)
{
str3 = Empty;
}
int length = ((str0.Length + str1.Length) + str2.Length) + str3.Length;
string dest = FastAllocateString(length);
FillStringChecked(dest, 0, str0);
FillStringChecked(dest, str0.Length, str1);
FillStringChecked(dest, str0.Length + str1.Length, str2);
FillStringChecked(dest, (str0.Length + str1.Length) + str2.Length, str3);
return dest;
}
private static string ConcatArray(string[] values, int totalLength)
{
string dest = FastAllocateString(totalLength);
int destPos = 0;
for (int i = 0; i < values.Length; i++)
{
FillStringChecked(dest, destPos, values[i]);
destPos += values[i].Length;
}
return dest;
}
private static unsafe void FillStringChecked(string dest, int destPos, string src)
{
int length = src.Length;
if (length > (dest.Length - destPos))
{
throw new IndexOutOfRangeException();
}
fixed (char* chRef = &dest.m_firstChar)
{
fixed (char* chRef2 = &src.m_firstChar)
{
wstrcpy(chRef + destPos, chRef2, length);
}
}
}
それで:
string what = "cat";
string inthehat = "The " + what + " in the hat!";
楽しみ!