百科问答小站 logo
百科问答小站 font logo



C#中的String.Format()这两种写法那个更好? 第1页

  

user avatar   rednaxelafx 网友的相关建议: 
      

这是个老话题了。

如果不想依赖CLR(或其它CLI VES的实现)中的优化的话,对于 int a; ,是 String.Format("a = {0}", a.ToString()) 比 String.Format("a = {0}", a) 的开销更小。

String.Format()有若干个重载版本,其中题主的例子会用到的是:

String.Format Method (String, Object)
       public static string Format(  string format,  object arg0 )      

所以如果直接传入a的话,int -> object需要做一次自动装箱(auto boxing),然后在 String.Format(string, object) 的内部实现里会再对这个object参数调用其ToString()方法来获得需要拼接的内容的String表示。

而如果传入a.ToString()的话,a.ToString()这个调用自身并不会导致装箱,得到的String传给String.Format()正好跟object匹配,就避免了一次额外的装箱操作。

具体到System.String在CoreCLR中的实现,String.Format(string, object) 的实现在

github.com/dotnet/corec

,它内部经过几层调用真正实现逻辑的地方在

github.com/dotnet/corec

,可以看到真正调用 arg.ToString() 的地方离 String.Format(string, object) 还隔着好几层,以CoreCLR的JIT(RyuJIT)的优化能力来说很难充分优化,于是就很难靠JIT优化自动消除 String.Format("a = {0}", a) 形式的代码导致的自动装箱。

但这个装箱的开销很大么?是否要教条式避免?

我个人是觉得这种地方显式调用ToString()是个好习惯,不会让代码丑很多而且可以自然地避免一些性能坑。

但同时我也不觉得这个地方的开销会很大,如果不调用ToString()真的导致很严重的问题的话profile的时候肯定会看到,看到的时候再改就是了。所以我完全不介意别人的习惯是在这种地方不显式调用ToString()。

有同学说这里调用a.ToString()可能会遇到NullReferenceException <- 请看清楚问题。题主的问题是假设a是int,是一个value type的情况下,是否应该显式调用a.ToString()再传给String.Format()。针对value type的方法调用是永远不会NRE的。

也有同学提到C# 6的新功能,interpolated strings。这当然是个好功能,我也很喜欢。这个功能的规范提到,当一个interpolated string被用在string类型的上下文中,它的语义等价于调用String.Format()。

请看Roslyn是如何翻译下面代码的:

Try Roslyn
       public class Test {     public string M(int a) {         return $"a = {a}";     } }      

会被目前的Roslyn翻译为等价于:

       public class Test {     public string M(int a)     {         return string.Format("a = {0}", a); // autobox a     } }      

也就是说C#的interpolated strings在当前Roslyn中的实现不会在调用String.Format()前自动给value type插入ToString()调用。这也是个很合理的设计——毕竟语言规范要尽可能简明扼要,把一个语法糖的解糖形式设计得简单也是好的。




  

相关话题

  .NET 下的性能问题如何定位? 
  如何解决Visual Studio Community 2015界面变黑的问题? 
  C#相对路径 连接Access数据库怎么写?? 
  如何提升ASP.NET网站首次打开速度? 
  C# 从语法角度比 Java 优秀在哪里? 
  .net内核能够移植到arm上,那windows能运行在上面么? 
  学 C# 的是不是被绑在 Windows 上? 
  .NET Core/.NET Framework 上的第三方常用类库/框架实现平均质量真的很差吗? 
  为什么C#的.NET库不默认提供「优先队列」容器? 
  C# 秒杀物品,程序如何控制并发? 

前一个讨论
bad apple怎么入侵广场上的led屏幕里的?
下一个讨论
千古帝王和牛顿、爱迪生、特斯拉这样的人对人类福祉影响谁更大?





© 2025-05-16 - tinynew.org. All Rights Reserved.
© 2025-05-16 - tinynew.org. 保留所有权利