百科问答小站 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()调用。这也是个很合理的设计——毕竟语言规范要尽可能简明扼要,把一个语法糖的解糖形式设计得简单也是好的。




  

相关话题

  Microsoft DLR现在是什么状态? 
  golang select 模式 在C#中如何实现? 
  C# ConcurrentQueue 怎么长时间循环取任务? 
  怎么在.net 平台上用和 WP7 开发类似的技术或思想来实现 Android 开发? 
  windows10 界面是用什么语言编写的?.NET在最新的操作系统中有哪些应用 
  为什么GO语言的字典性能不如C#? 
  C#每个类代码一大坨,有什么好的方能展示他的公有方法和属性? 
  关于 C# 中「string + int」的执行机制的疑惑? 
  如何判断 string 是否为合法的 C# 变量名? 
  .net程序卡死是不是和修复的漏洞有关系? 

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





© 2024-11-13 - tinynew.org. All Rights Reserved.
© 2024-11-13 - tinynew.org. 保留所有权利