-
Notifications
You must be signed in to change notification settings - Fork 251
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
feat: Remove default collection initialization for perf reasons #2284
base: main
Are you sure you want to change the base?
Conversation
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.
For those sets of changes, it's always better to do it at one place with a draft, have a conversation, and then replicate all over.
Working on reverting these changes to remove default collection initialization altogether. There are lots of failing tests due to null reference exceptions but I'm working on fixing them |
…ents' into feat/memory-perf-improvements
@@ -115,7 +115,7 @@ public void ValidateDefaultShouldNotHaveDataTypeMismatchForComplexSchema() | |||
var schema = new OpenApiSchema | |||
{ | |||
Type = JsonSchemaType.Object, | |||
Properties = | |||
Properties = new Dictionary<string, IOpenApiSchema> |
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.
Does the shortcut syntax work where you only put "new" and the type is inferred?
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.
No that doesn't work.
Properties is an implementation of the IDictionary
interface so you have to explicitly define the concrete type here.
public IDictionary<string, IOpenApiSchema>? Properties { get; set; } = new Dictionary<string, IOpenApiSchema>(StringComparer.Ordinal); |
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.
but we could change the property type as well. (see my latest comment)
Looks like I missed this conversation. Is there a summary of why we want to force users to initialize these collections manually? Is ASP.NET team ok with this change? |
If we allocate by default with field initialization (current implementation), we allocate the memory when the object gets created. If we lazy initialize the collection:
I believe the reason why we allocated on initialization historically was because:
In addition to all that, in a lot of cases, the consumers end up doing object initialization syntax, which means they are initializing the collections anyway. For all those reasons I suggest we stick to the idiomatic expressions of the platform, and leave it null, letting the compiler help consumers while reducing memory usage. |
cc: @captainsafia |
I had not considered this. And I don't think we should change deserialization to leverage existing collections.
I think most of the time that isn't an important distinction to make. I think there are only a couple of collections in the model where that is significant. e.g. Paths and security requirements.
The reason was so that a consumer editing/creating a document did not have instantiate a type that they may or may not know what that type is. e.g. sometimes we use dictionary<string,foo> sometimes we have dedicated classes OpenAPIResponses. Eight years ago the object initialization syntax was not as developed and wasn't used nearly as much.
Is it? I agree that C# has been introducing new ways to initialize objects/arrays/collections over the past few years, but is there some official design guidance that this is idiomatic? Anyway, I think because the support for object initialization is better and because we want to optimize perf for the common case of deserialization, and because we now have NRT, I am more comfortable moving to a model where we don't lazy instantiate. |
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 great progress!
One thing I also wanted us to discuss but forgot to mention earlier: Why are collections returning their interface type? What benefit does it provide? the obvious drawback is that it prevents from using the collection initialization syntax []
, especially for dictionaries as we can see in the unit tests. CC @darrelmiller
I'm not sure what the initial intention was for defining the dictionaries as interface types instead of concrete types. |
I suspect the reason we implemented these properties as interfaces was to leave the possibility of changing the concrete implementation if necessary. I don't have any objection to changing to the concrete classes now there is a clear benefit. |
|
Fixes #1971