为什么我的 Func 如此之慢?
生活随笔
收集整理的這篇文章主要介紹了
为什么我的 Func 如此之慢?
小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
咨詢區(qū)
Ricky G:
我正在給項(xiàng)目做性能和代碼優(yōu)化,比如將重復(fù)的代碼提煉成到一個(gè)可重用的方法中,為了能夠達(dá)到可重用目的,我用 Func<T> 作為方法參數(shù)。
public?int?Calculate(Func<int>?expr){return?expr();}當(dāng)我用 Benchmark 對(duì)優(yōu)化前和優(yōu)化后的代碼做基準(zhǔn)測(cè)試時(shí),居然發(fā)現(xiàn)優(yōu)化后的在性能上慢了兩倍?
完整的代碼如下:
using?System; using?System.Runtime.CompilerServices; using?BenchmarkDotNet.Attributes; using?BenchmarkDotNet.Exporters; using?BenchmarkDotNet.Loggers; using?BenchmarkDotNet.Running;namespace?ConsoleApp1 {class?Program{static?void?Main(string[]?args){var?summary?=?BenchmarkRunner.Run<Calculations>();var?logger?=?ConsoleLogger.Default;MarkdownExporter.Console.ExportToLog(summary,?logger);Console.WriteLine(summary);}}public?class?Calculations{public?Random?RandomGeneration?=?new?Random();[Benchmark]public?void?CalculateNormal(){var?s?=??RandomGeneration.Next()?*?RandomGeneration.Next();}[Benchmark]public?void?CalculateUsingFunc(){Calculate(()?=>?RandomGeneration.Next()?*?RandomGeneration.Next());}[MethodImpl(MethodImplOptions.AggressiveInlining)]public?int?Calculate(Func<int>?expr){return?expr();}} }回答區(qū)
Jon Skeet:
慢兩倍的真正原因在于你的每一次調(diào)用都會(huì)生成一個(gè)新的 delegate,所以我一點(diǎn)都不感到驚訝。
如果你的 lambda 表達(dá)式不需要捕獲 this 或者其他 local 變量,那你完全可以將其提取成一個(gè)單例形式,這樣就可以避免每一次都調(diào)用new了,開銷自然就大大減少了。
下面是我修改后的代碼:
public?class?Calculations {public?Random?RandomGeneration?=?new?Random();private?Func<int>?exprField;public?Calculations(){exprField?=?()?=>?RandomGeneration.Next()?*?RandomGeneration.Next();}[Benchmark]public?void?CalculateNormal(){var?s?=??RandomGeneration.Next()?*?RandomGeneration.Next();}[Benchmark]public?void?CalculateUsingFunc(){Calculate(()?=>?RandomGeneration.Next()?*?RandomGeneration.Next());}[Benchmark]public?void?CalculateUsingFuncField(){Calculate(exprField);}[MethodImpl(MethodImplOptions.AggressiveInlining)]public?int?Calculate(Func<int>?expr){return?expr();} }benchmark 結(jié)果。
|??????????????????Method?|?????Mean?|????Error?|???StdDev?| |------------------------?|---------:|---------:|---------:| |?????????CalculateNormal?|?27.61?ns?|?0.438?ns?|?0.388?ns?| |??????CalculateUsingFunc?|?48.74?ns?|?1.009?ns?|?0.894?ns?| |?CalculateUsingFuncField?|?32.53?ns?|?0.698?ns?|?0.717?ns?|雖然還是有一些差距,但相比你的方式已經(jīng)大大改善了。
點(diǎn)評(píng)區(qū)
這個(gè)確實(shí)是在寫代碼的過程中容易忽視的地方,學(xué)習(xí)了,用 Benchmark 做基準(zhǔn)測(cè)試也是值得大家了解的。
總結(jié)
以上是生活随笔為你收集整理的为什么我的 Func 如此之慢?的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 基于 gRPC 和 .NET Core
- 下一篇: Hello Blazor:(8)启用深色