Skip to content
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

Add documentation for CA1864. #36059

Merged
merged 7 commits into from
Jul 7, 2023
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
94 changes: 94 additions & 0 deletions docs/fundamentals/code-analysis/quality-rules/ca1864.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
---
title: "CA1864: Prefer the 'IDictionary.TryAdd(TKey, TValue)' method"
description: "Learn about code analyzer rule CA1864 - Prefer the 'IDictionary.TryAdd(TKey, TValue)' method"
ms.date: 06/30/2023
ms.topic: reference
f1_keywords:
- CA1864
- PreferDictionaryTryMethodsOverContainsKeyGuardAnalyzer
helpviewer_keywords:
- CA1864
dev_langs:
- CSharp
- VB
---

# CA1864: Prefer the 'IDictionary.TryAdd(TKey, TValue)' method

| | Value |
| ----------------------------------- |----------------------------------------|
| **Rule ID** | CA1864 |
| **Category** | [Performance](performance-warnings.md) |
| **Fix is breaking or non-breaking** | Non-breaking |
| **Enabled by default in .NET 7** | No |

## Cause

<xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> is guarded by on a <xref:System.Collections.Generic.Dictionary%602.ContainsKey(%600)?displayProperty=nameWithType> call.
gewarren marked this conversation as resolved.
Show resolved Hide resolved

## Rule description

Both <xref:System.Collections.Generic.Dictionary%602.ContainsKey(%600)?displayProperty=nameWithType> and <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> perform a lookup of the value. <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> also throws an exception if the key is already present in the dictionary. It is more efficient to call <xref:System.Collections.Generic.Dictionary%602.TryAdd(%600)?displayProperty=nameWithType> which will return a `bool` indicating if the value was added or not. It will not override the key's value if the key is already present.
gewarren marked this conversation as resolved.
Show resolved Hide resolved

## How to fix violations

Replace a call to <xref:System.Collections.Generic.Dictionary%602.ContainsKey(%600)?displayProperty=nameWithType> followed by <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> with a call to <xref:System.Collections.Generic.Dictionary%602.TryAdd(%600)?displayProperty=nameWithType>.
gewarren marked this conversation as resolved.
Show resolved Hide resolved

## Example

The following code snippet shows a violation of CA1864:

```csharp
void Run(IDictionary<int, string> dictionary)
{
if(!dictionary.ContainsKey(2)) {
dictionary.Add(2, "Hello World");
}
}
```

```vb
Sub Run(dictionary As IDictionary(Of Integer, String))
If Not dictionary.ContainsKey(2) Then
dictionary.Add(2, "Hello World")
End If
End Sub
```

The following code snippet fixes the violation:

```csharp
void Run(IDictionary<int, string> dictionary)
{
dictionary.TryAdd(2, "Hello World");
}
```

```vb
Sub Run(dictionary As IDictionary(Of Integer, String))
dictionary.TryAdd(2, "Hello World")
End Sub
```

## When to suppress warnings

It's safe to suppress this warning if performance isn't a concern and when the exception potentially thrown by <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> is caught.
gewarren marked this conversation as resolved.
Show resolved Hide resolved

## Suppress a warning

If you just want to suppress a single violation, add preprocessor directives to your source file to disable and then re-enable the rule.

```csharp
#pragma warning disable CA1864
// The code that's violating the rule is on this line.
#pragma warning restore CA1864
```

To disable the rule for a file, folder, or project, set its severity to `none` in the [configuration file](../configuration-files.md).

```ini
[*.{cs,vb}]
dotnet_diagnostic.CA1864.severity = none
```

For more information, see [How to suppress code analysis warnings](../suppress-warnings.md).
1 change: 1 addition & 0 deletions docs/fundamentals/code-analysis/quality-rules/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,7 @@ The following table lists code quality analysis rules.
> | [CA1859: Use concrete types when possible for improved performance](ca1859.md) | Code uses interface types or abstract types, leading to unnecessary interface calls or virtual calls. |
> | [CA1860: Avoid using 'Enumerable.Any()' extension method](ca1860.md) | It's more efficient and clearer to use `Length`, `Count`, or `IsEmpty` (if possible) than to call <xref:System.Linq.Enumerable.Any%2A?displayProperty=nameWithType> to determine whether a collection type has any elements. |
> | [CA1861: Avoid constant arrays as arguments](ca1861.md) | Constant arrays passed as arguments are not reused which implies a performance overhead. Consider extracting them to 'static readonly' fields to improve performance. |
> | [CA1864: Prefer the 'IDictionary.TryAdd(TKey, TValue)' method](ca1864.md) | Both <xref:System.Collections.Generic.Dictionary%602.ContainsKey(%600)?displayProperty=nameWithType> and <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> perform a lookup of the value. <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> also throws an exception if the key is already present in the dictionary. It is more efficient to call <xref:System.Collections.Generic.Dictionary%602.TryAdd(%600)?displayProperty=nameWithType> which will return a `bool` indicating if the value was added or not. It will not override the key's value if the key is already present. |
gewarren marked this conversation as resolved.
Show resolved Hide resolved
> | [CA2000: Dispose objects before losing scope](ca2000.md) | Because an exceptional event might occur that will prevent the finalizer of an object from running, the object should be explicitly disposed before all references to it are out of scope. |
> | [CA2002: Do not lock on objects with weak identity](ca2002.md) |An object is said to have a weak identity when it can be directly accessed across application domain boundaries. A thread that tries to acquire a lock on an object that has a weak identity can be blocked by a second thread in a different application domain that has a lock on the same object. |
> | [CA2007: Do not directly await a Task](ca2007.md) | An asynchronous method [awaits](../../../csharp/language-reference/operators/await.md) a <xref:System.Threading.Tasks.Task> directly. When an asynchronous method awaits a <xref:System.Threading.Tasks.Task> directly, continuation occurs in the same thread that created the task. This behavior can be costly in terms of performance and can result in a deadlock on the UI thread. Consider calling <xref:System.Threading.Tasks.Task.ConfigureAwait(System.Boolean)?displayProperty=nameWithType> to signal your intention for continuation. |
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,3 +70,4 @@ Performance rules support high-performance libraries and applications.
| [CA1859: Use concrete types when possible for improved performance](ca1859.md) | Code uses interface types or abstract types, leading to unnecessary interface calls or virtual calls. |
| [CA1860: Avoid using 'Enumerable.Any()' extension method](ca1860.md) | It's more efficient and clearer to use `Length`, `Count`, or `IsEmpty` (if possible) than to call <xref:System.Linq.Enumerable.Any%2A?displayProperty=nameWithType> to determine whether a collection type has any elements. |
| [CA1861: Avoid constant arrays as arguments](ca1861.md) | Constant arrays passed as arguments are not reused which implies a performance overhead. Consider extracting them to 'static readonly' fields to improve performance. |
| [CA1864: Prefer the 'IDictionary.TryAdd(TKey, TValue)' method](ca1864.md) | Both <xref:System.Collections.Generic.Dictionary%602.ContainsKey(%600)?displayProperty=nameWithType> and <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> perform a lookup of the value. <xref:System.Collections.Generic.Dictionary%602.Add(%600)?displayProperty=nameWithType> also throws an exception if the key is already present in the dictionary. It is more efficient to call <xref:System.Collections.Generic.Dictionary%602.TryAdd(%600)?displayProperty=nameWithType> which will return a `bool` indicating if the value was added or not. It will not override the key's value if the key is already present. |
gewarren marked this conversation as resolved.
Show resolved Hide resolved
12 changes: 7 additions & 5 deletions docs/navigate/tools-diagnostics/toc.yml
Original file line number Diff line number Diff line change
Expand Up @@ -349,7 +349,7 @@ items:
- name: Specialized diagnostics
items:
- name: Overview
href: ../../core/diagnostics/specialized-diagnostics-overview.md
href: ../../core/diagnostics/specialized-diagnostics-overview.md
- name: Event Source
items:
- name: Overview
Expand Down Expand Up @@ -386,7 +386,7 @@ items:
- name: Type-system events
href: ../../fundamentals/diagnostics/runtime-type-events.md
- name: Collect diagnostics in containers
href: ../../core/diagnostics/diagnostics-in-containers.md
href: ../../core/diagnostics/diagnostics-in-containers.md
- name: Dumps
items:
- name: Overview
Expand All @@ -401,9 +401,9 @@ items:
- name: Collect dumps on crash
href: ../../core/diagnostics/collect-dumps-crash.md
- name: Symbols
href: ../../core/diagnostics/symbols.md
href: ../../core/diagnostics/symbols.md
- name: EventPipe
href: ../../core/diagnostics/eventpipe.md
href: ../../core/diagnostics/eventpipe.md
- name: Diagnostic port
items:
- name: Overview
Expand All @@ -418,7 +418,7 @@ items:
- name: DiagnosticSource and DiagnosticListener
items:
- name: Get started
href: ../../core/diagnostics/diagnosticsource-diagnosticlistener.md
href: ../../core/diagnostics/diagnosticsource-diagnosticlistener.md
- name: .NET CLI global tools
items:
- name: Overview
Expand Down Expand Up @@ -801,6 +801,8 @@ items:
href: ../../fundamentals/code-analysis/quality-rules/ca1860.md
- name: CA1861
href: ../../fundamentals/code-analysis/quality-rules/ca1861.md
- name: CA1864
href: ../../fundamentals/code-analysis/quality-rules/ca1864.md
- name: SingleFile rules
items:
- name: Overview
Expand Down