From 41991e6090c58f3793a59cb37cd89126044211c9 Mon Sep 17 00:00:00 2001 From: Sean Wheeler Date: Thu, 5 Dec 2024 09:46:05 -0600 Subject: [PATCH 1/2] Add perf comparisons for creating new objects (#11561) --- .../script-authoring-considerations.md | 63 ++++++++++++++++++- 1 file changed, 62 insertions(+), 1 deletion(-) diff --git a/reference/docs-conceptual/dev-cross-plat/performance/script-authoring-considerations.md b/reference/docs-conceptual/dev-cross-plat/performance/script-authoring-considerations.md index 4b1dee0358b2..3012f51a91b6 100644 --- a/reference/docs-conceptual/dev-cross-plat/performance/script-authoring-considerations.md +++ b/reference/docs-conceptual/dev-cross-plat/performance/script-authoring-considerations.md @@ -1,6 +1,6 @@ --- description: Scripting for Performance in PowerShell -ms.date: 05/07/2024 +ms.date: 12/05/2024 title: PowerShell scripting performance considerations --- @@ -627,6 +627,67 @@ Unwrapped = 42.92 ms The unwrapped example is **372 times faster**. Also, notice that the first implementation requires the **Append** parameter, which isn't required for the later implementation. +## Object creation + +Creating objects using the `New-Object` cmdlet can be slow. The following code compares the +performance of creating objects using the `New-Object` cmdlet to the `[pscustomobject]` type +accelerator. + +```powershell +Measure-Command { + $test = 'PSCustomObject' + for ($i = 0; $i -lt 100000; $i++) { + $resultObject = [PSCustomObject]@{ + Name = 'Name' + Path = 'FullName' + } + } +} | Select-Object @{n='Test';e={$test}},TotalSeconds + +Measure-Command { + $test = 'New-Object' + for ($i = 0; $i -lt 100000; $i++) { + $resultObject = New-Object -TypeName PSObject -Property @{ + Name = 'Name' + Path = 'FullName' + } + } +} | Select-Object @{n='Test';e={$test}},TotalSeconds +``` + +```Output +Test TotalSeconds +---- ------------ +PSCustomObject 0.48 +New-Object 3.37 +``` + +PowerShell 5.0 added the `new()` static method for all .NET types. The following code compares the +performance of creating objects using the `New-Object` cmdlet to the `new()` method. + +```powershell +Measure-Command { + $test = 'new() method' + for ($i = 0; $i -lt 100000; $i++) { + $sb = [System.Text.StringBuilder]::new(1000) + } +} | Select-Object @{n='Test';e={$test}},TotalSeconds + +Measure-Command { + $test = 'New-Object' + for ($i = 0; $i -lt 100000; $i++) { + $sb = New-Object -TypeName System.Text.StringBuilder -ArgumentList 1000 + } +} | Select-Object @{n='Test';e={$test}},TotalSeconds +``` + +```Output +Test TotalSeconds +---- ------------ +new() method 0.59 +New-Object 3.17 +``` + ## Use OrderedDictionary to dynamically create new objects There are situations where we may need to dynamically create objects based on some input, the From 4b8a7406c4302af111848fbcab7dbb81b74304f3 Mon Sep 17 00:00:00 2001 From: Sean Wheeler Date: Thu, 5 Dec 2024 10:11:12 -0600 Subject: [PATCH 2/2] Add performance update (#11562) --- .../whats-new/What-s-New-in-PowerShell-75.md | 98 ++++++++++++++++++- 1 file changed, 97 insertions(+), 1 deletion(-) diff --git a/reference/docs-conceptual/whats-new/What-s-New-in-PowerShell-75.md b/reference/docs-conceptual/whats-new/What-s-New-in-PowerShell-75.md index 1018726fa338..6faf74fe3c19 100644 --- a/reference/docs-conceptual/whats-new/What-s-New-in-PowerShell-75.md +++ b/reference/docs-conceptual/whats-new/What-s-New-in-PowerShell-75.md @@ -1,7 +1,7 @@ --- title: What's New in PowerShell 7.5 description: New features and changes released in PowerShell 7.5 -ms.date: 11/18/2024 +ms.date: 12/05/2024 --- # What's New in PowerShell 7.5 @@ -151,6 +151,102 @@ The following experimental features are included in PowerShell 7.5-rc.1: - [PSSerializeJSONLongEnumAsNumber][06] - `ConvertTo-Json` now treats large enums as numbers ([#20999][20999]) (Thanks @jborean93!) +## Performance improvements + +PowerShell 7.5-rc.1 included [PR#23901][23901] from @jborean93 that improves the performance of the +`+=` operation for an array of objects. + +The following example measures the performance for different methods of adding elements to an array. + +```powershell +$tests = @{ + 'Direct Assignment' = { + param($count) + + $result = foreach($i in 1..$count) { + $i + } + } + 'List.Add(T)' = { + param($count) + + $result = [Collections.Generic.List[int]]::new() + foreach($i in 1..$count) { + $result.Add($i) + } + } + 'Array+= Operator' = { + param($count) + + $result = @() + foreach($i in 1..$count) { + $result += $i + } + } +} + +5kb, 10kb | ForEach-Object { + $groupResult = foreach($test in $tests.GetEnumerator()) { + $ms = (Measure-Command { & $test.Value -Count $_ }).TotalMilliseconds + +[pscustomobject]@{ + CollectionSize = $_ + Test = $test.Key + TotalMilliseconds = [math]::Round($ms, 2) + } + +[GC]::Collect() + [GC]::WaitForPendingFinalizers() + } + +$groupResult = $groupResult | Sort-Object TotalMilliseconds + $groupResult | Select-Object *, @{ + Name = 'RelativeSpeed' + Expression = { + $relativeSpeed = $_.TotalMilliseconds / $groupResult[0].TotalMilliseconds + $speed = [math]::Round($relativeSpeed, 2).ToString() + 'x' + if ($speed -eq '1x') { $speed } else { $speed + ' slower' } + } + } | Format-Table -AutoSize +} +``` + +When you run the script in PowerShell 7.4.6, you see that using the `+=` operator is the slowest +method. + +```Output +CollectionSize Test TotalMilliseconds RelativeSpeed +-------------- ---- ----------------- ------------- + 5120 Direct Assignment 4.17 1x + 5120 List.Add(T) 90.79 21.77x slower + 5120 Array+= Operator 342.58 82.15x slower + + +CollectionSize Test TotalMilliseconds RelativeSpeed +-------------- ---- ----------------- ------------- + 10240 Direct Assignment 0.64 1x + 10240 List.Add(T) 184.10 287.66x slower + 10240 Array+= Operator 1668.13 2606.45x slower +``` + +When you run the script in PowerShell 7.5-rc.1, you see that using the `+=` operator is much faster +than PowerShell 7.4.6. Now, it's also faster than using the `List.Add(T)` method. + +```Output +CollectionSize Test TotalMilliseconds RelativeSpeed +-------------- ---- ----------------- ------------- + 5120 Direct Assignment 4.71 1x + 5120 Array+= Operator 40.42 8.58x slower + 5120 List.Add(T) 92.17 19.57x slower + + +CollectionSize Test TotalMilliseconds RelativeSpeed +-------------- ---- ----------------- ------------- + 10240 Direct Assignment 1.76 1x + 10240 Array+= Operator 104.73 59.51x slower + 10240 List.Add(T) 173.00 98.3x slower +``` + [chg]: https://github.com/PowerShell/PowerShell/blob/master/CHANGELOG/preview.md