-
Notifications
You must be signed in to change notification settings - Fork 352
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Avoid creating unused objects when calling ODataUri.Clone() #2188
Avoid creating unused objects when calling ODataUri.Clone() #2188
Conversation
This PR has Quantification details
Why proper sizing of changes matters
Optimal pull request sizes drive a better predictable PR flow as they strike a
What can I do to optimize my changes
How to interpret the change counts in git diff output
Was this comment helpful? 👍 :ok_hand: :thumbsdown: (Email) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fix loos good to me. Lazy initialization in the getter minimizes the changes and given this class is not meant to be used concurrently it works just fine. It also improves any scenarios where ODataUri is created, but ParameterAliasValueAccessor is never used.
@@ -30,13 +30,13 @@ public sealed class ODataUri | |||
/// </summary> | |||
private Uri serviceRoot; | |||
|
|||
private ParameterAliasValueAccessor parameterAliasValueAccessor; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Line 130, ParameterAliasNodes has a null check on the property. That null check does not seem required since it will always evaluate true? Or if it is, it should be replaced with the field, instead of the property?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is for the ParameterAliasNodes getter - it has a null check for ParameterAliasValueAccessor which is redundant in the current iteration, since ParameterAliasValueAccessor is never null. Hence should check against the field instead of the property, or perhaps remove the null check.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Moving initialization from constructor to getter looks fine . Can you see how much % improvement we got . Let me know if you want me to run a local benchmark test
If we do this kind of work more often I don't think it is good to wait for deployment and measurement via Graph. I would recommend writing benchmark tests (one sdk should be sufficient) and compare the before and after. graph measurements are a moving target and it takes a long time to collect that information. |
{ | ||
get | ||
{ | ||
if (parameterAliasValueAccessor == null) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a change of behaviour - could be for the better, but still a change :-) - and there probably should be a test to expect this current behaviour...
- Previously: value is set in ctor, you could then be able to
set
a null to that value and a subsequentget
would give null. - Currently: even if you ever
set
a null value, you would neverget
it
It could be that current behaviour is fixing unwanted null gets but yeah, probably a test to document this...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmm, good point
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would it be possible to change the setter from internal to private ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Even the getter, does not seem it needs to be public/internal as of now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes; the setter is currently internal and only called when initialized to a non-null value in the constructor, so we should be able to assume it is never null. We can change the setter (and getter) to private and change the null check on 128-133 with an non-null assert.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Issues
This pull request (partially) addresses #2163 .
Description
The fix
ODataUri.Clone()
creates a new instance ofODataUri
usingnew ODataUri()
then shallow copies the properties (references) from the "old"ODataUri
to the new one. However, the defaultODataUri
constructor creates a new instance ofDictionary<string, string>
and a new instance ofParameterAliasValueAccessor
. TheODataUri.ParameterAliasValueaAccessor
property is immediately replaced byODataUri.Clone()
with the reference from the old uri. So the instances created in the default constructor are a waste of memory and unnecessary allocation/GC overhead.I fixed this issue by creating the
ParameterAliasValueAccessor
instance lazily in the property getter. I thought this to be the safest solution, below are other solutions that I considered:Clone()
, and I did not have time to investigate all those usage scenariosParameterAliasValueAccessor
) that are then directly passed to properties. This constructor seems close to whatODataUri.Clone
is doing, except that when setting theCustomQueryOptions
property:this.CustomQueryOptions = new ReadOnlyCollection<QueryNode>(customQueryOptions.ToList());
, where as inODataUri.Clone()
the property reference is simply copied without creating any new object. So this would not be a suitable substitute.Expected impact
While this change does not reduce the number of
ODataUri
instances allocated (which is what the original issue pointed to in the screenshot), it does significantly reduce the number ofDictionary
instances created as a result ofODataUri.Clone()
.This CPR screenshot shows that
data:image/s3,"s3://crabby-images/913d0/913d0a21e8ba2f4a87ba5ce09ae05f6f9c24b52a" alt="image"
Dictionary<string, string>
created fromODataUri.Clone()
account for and 1.3% (0.69 + 0.61) of allocated size:The following screenshot shows
data:image/s3,"s3://crabby-images/8678a/8678a30ed51606d967b0a152419a97527870ea58" alt="AGS SingleValueNode dict allocation size"
Dictionary<string, SingleValueNode>
fromODataUri.Clone()
account for 0.65%.The following screenshot shows
ParameterAliasValueAccessor
fromODataUri.Clone()
accounts for 0.23%So in total, there's up to 2.18% that we can potentially shed off. This is based on the assumptions that the sizes indicated in the screenshots do not include sizes of referenced objects (otherwise
ParameterAliasValueAccessor
would have a larger inclusive size than the dictionaries it contains).I've done some local profiling before and after the fix to get a better estimate of how much we can expect to reduce. The profiling were based on running a simple service that uses that OData writer to write a response of 5000 entities (base on this experments project) and collecting allocation data using the .NET Object Allocation Tracker tool that's part of the Visual Studio Performance Profiler toolset.
The following estimates may not hold true in production given the difference in usage patterns and data, but it may be a good idea to compare the results once we measure them in production with the estimates here later on. I think that will help us make better estimates over time.
Impact on number of allocations
The first 2 screenshots below show the total allocations from
ODataUri.Clone()
from my local profile before and after this change. You can see total allocations dropped from 175k to 35k, i.e. about 140k drop in allocations, which is about 80%.ODataUri.Clone()
accounts in total for 4.13% of allocations (inclusive), 80% drop would should shed off about 3.3% of allocations in AGS assuming the experiment data is a reflection of what happens in production.Impact on allocation size
Impact on allocation size is trickier to estimate cause the .NET allocation tracker in VS doesn't display the inclusive allocation size for a given type. I'm also not sure how the inclusive size is computed in CPR (on the stack tab, both inclusive and exclusive columns have the same value), on the Frames tab, they have different values, but this tab doesn't show the breakdown of the types allocated by a function.
That said, I thin we can realistically estimate that at least all
Dictionary<string, string>
,Dictionary<string, SingleValueNode>
andParameterAliasValueAccessor
allocated fromODataUri.Clone()
will be eliminated, and I've already calculated that above to be 2.18% of total allocated bytes.Below are some screenshots from my local profile showing the allocated size of
Dictionary
instances before and after the change.Impact on collection
On my local profile, there were 3 fewer GC collections after the change. But I'm not sure if this is reliable.
Checklist (Uncheck if it is not completed)
Additional work necessary
If documentation update is needed, please add "Docs Needed" label to the issue and provide details about the required document change in the issue.