ToStringEnumerable.cs 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. using System.Text;
  2. using BenchmarkDotNet.Attributes;
  3. using Tui = Terminal.Gui;
  4. namespace Terminal.Gui.Benchmarks.Text.StringExtensions;
  5. /// <summary>
  6. /// Benchmarks for <see cref="Tui.StringExtensions.ToString(IEnumerable{Rune})"/> performance fine-tuning.
  7. /// </summary>
  8. [MemoryDiagnoser]
  9. public class ToStringEnumerable
  10. {
  11. /// <summary>
  12. /// Benchmark for previous implementation.
  13. /// </summary>
  14. [Benchmark]
  15. [ArgumentsSource (nameof (DataSource))]
  16. public string Previous (IEnumerable<Rune> runes, int len)
  17. {
  18. return StringConcatInLoop (runes);
  19. }
  20. /// <summary>
  21. /// Benchmark for current implementation with char buffer and
  22. /// fallback to rune chars appending to StringBuilder.
  23. /// </summary>
  24. /// <param name="runes"></param>
  25. /// <returns></returns>
  26. [Benchmark (Baseline = true)]
  27. [ArgumentsSource (nameof (DataSource))]
  28. public string Current (IEnumerable<Rune> runes, int len)
  29. {
  30. return Tui.StringExtensions.ToString (runes);
  31. }
  32. /// <summary>
  33. /// Previous implementation with string concatenation in a loop.
  34. /// </summary>
  35. private static string StringConcatInLoop (IEnumerable<Rune> runes)
  36. {
  37. var str = string.Empty;
  38. foreach (Rune rune in runes)
  39. {
  40. str += rune.ToString ();
  41. }
  42. return str;
  43. }
  44. public IEnumerable<object []> DataSource ()
  45. {
  46. // Extra length argument as workaround for the summary grouping
  47. // different length collections to same baseline making comparison difficult.
  48. foreach (string text in GetTextData ())
  49. {
  50. Rune [] runes = [..text.EnumerateRunes ()];
  51. yield return [runes, runes.Length];
  52. }
  53. }
  54. private IEnumerable<string> GetTextData ()
  55. {
  56. string textSource =
  57. """
  58. Ĺόŕéḿ íṕśúḿ d́όĺόŕ śít́ áḿét́, ćόńśéćt́ét́úŕ ád́íṕíśćíńǵ éĺít́. Ṕŕáéśéńt́ q́úíś ĺúćt́úś éĺít́. Íńt́éǵéŕ út́ áŕćú éǵét́ d́όĺόŕ śćéĺéŕíśq́úé ḿát́t́íś áć ét́ d́íáḿ.
  59. Ṕéĺĺéńt́éśq́úé śéd́ d́áṕíb́úś ḿáśśá, v́éĺ t́ŕíśt́íq́úé d́úí. Śéd́ v́ít́áé ńéq́úé éú v́éĺít́ όŕńáŕé áĺíq́úét́. Út́ q́úíś όŕćí t́éḿṕόŕ, t́éḿṕόŕ t́úŕṕíś íd́, t́éḿṕúś ńéq́úé.
  60. Ṕŕáéśéńt́ śáṕíéń t́úŕṕíś, όŕńáŕé v́éĺ ḿáúŕíś át́, v́áŕíúś śúśćíṕít́ áńt́é. Út́ ṕúĺv́íńáŕ t́úŕṕíś ḿáśśá, q́úíś ćúŕśúś áŕćú f́áúćíb́úś íń.
  61. Óŕćí v́áŕíúś ńát́όq́úé ṕéńát́íb́úś ét́ ḿáǵńíś d́íś ṕáŕt́úŕíéńt́ ḿόńt́éś, ńáśćét́úŕ ŕíd́íćúĺúś ḿúś. F́úśćé át́ éx́ b́ĺáńd́ít́, ćόńv́áĺĺíś q́úáḿ ét́, v́úĺṕút́át́é ĺáćúś.
  62. Śúśṕéńd́íśśé śít́ áḿét́ áŕćú út́ áŕćú f́áúćíb́úś v́áŕíúś. V́ív́áḿúś śít́ áḿét́ ḿáx́íḿúś d́íáḿ. Ńáḿ éx́ ĺéό, ṕh́áŕét́ŕá éú ĺόb́όŕt́íś át́, t́ŕíśt́íq́úé út́ f́éĺíś.
  63. """;
  64. int[] lengths = [1, 10, 100, textSource.Length / 2, textSource.Length];
  65. foreach (int length in lengths)
  66. {
  67. yield return textSource [..length];
  68. }
  69. string textLongerThanStackallocThreshold = string.Concat(Enumerable.Repeat(textSource, 10));
  70. yield return textLongerThanStackallocThreshold;
  71. }
  72. }