编译器可以在字符串上优化ToString()吗?

本文关键字:ToString 优化 字符串 编译器 | 更新日期: 2023-09-27 18:17:05

我相信每个人都遇到过喜欢 ToString()方法的开发人员。我们可能都见过类似下面的代码:

public static bool CompareIfAnyMatchesOrHasEmpty(List<string> list1, List<string> list2)
{
    bool result = false;
    foreach (string item1 in list1)
    {
        foreach (string item2 in list2)
        {
            if (item1.ToString() == item2.ToString())
            {
                result = true;
            }
            if (item1.ToString() == "")
            {
                result = true;
            }
        }
    }
    return result;
}

我想知道的是,如果ToString()方法(空的,没有格式化的一个)可以被编译器优化掉?我的假设是它没有,因为它最初是在object上定义的。因此,我提出了第二个问题,即清除此类实例的任何努力是否值得?

编译器可以在字符串上优化ToString()吗?

c#编译器不会对其进行优化。然而,在运行时,我相信这可能会被CLR中的JIT编译器内联,因为string.ToString()只返回自己。

String.ToString甚至与TargetedPatchingOptOutAttribute一起声明,这允许它在从其他程序集调用时被NGEN内联,所以它显然是一个内联目标。

它当然可以被编译器优化掉,但是它们可能不会,因为它是微不足道的。在决定是否值得进行任何优化之前,请先进行一些测试。让我们试试吧!

List<string> strings = Enumerable.Range(1, 10000000).Select(x => Guid.NewGuid().ToString()).ToList();
var sw= Stopwatch.StartNew();
foreach (var str in strings) {
    if (!str.ToString().Equals(str.ToString())) {
        throw new ApplicationException("The world is ending");
    }
}
sw.Stop();
Console.WriteLine("Took: " + sw.Elapsed.TotalMilliseconds);
sw = Stopwatch.StartNew();
foreach (var str in strings) {
    if (!str.Equals(str)) {
        throw new ApplicationException("The world is ending");
    }
}
sw.Stop();
Console.WriteLine("Took: " + sw.Elapsed.TotalMilliseconds);

好的,我们在一个有1000万个元素的循环中。与非tostring版本相比,tostring(调用两次)版本需要多长时间?

这是我的机器上的文件:

Took: 261.6189 
Took: 231.2615

所以,是的。我在1000万次迭代中节省了30毫秒。所以…是的,我会说不,不值得。在所有。

现在,代码是否应该因为愚蠢而更改?是的。我的论点是,"这是不必要的,让我认为这不是一个字符串一眼。这需要我的大脑循环来处理,而且毫无意义。别这么做。"不要从优化的角度来争论。