AspNetCore.Docs/aspnetcore/performance/caching/distributed.md

150 lines
8.9 KiB
Markdown
Raw Normal View History

2016-10-29 01:35:15 +08:00
---
title: Work with a distributed cache in ASP.NET Core
2016-11-12 00:28:14 +08:00
author: ardalis
description: Learn how to use ASP.NET Core distributed caching to improve app performance and scalability, especially in a cloud or server farm environment.
2018-01-29 23:21:31 +08:00
ms.author: riande
2018-06-05 01:03:05 +08:00
ms.custom: mvc
ms.date: 02/14/2017
2016-10-29 01:35:15 +08:00
uid: performance/caching/distributed
---
# Work with a distributed cache in ASP.NET Core
2016-10-29 01:35:15 +08:00
By [Steve Smith](https://ardalis.com/)
2016-10-29 01:35:15 +08:00
Distributed caches can improve the performance and scalability of ASP.NET Core apps, especially when hosted in a cloud or server farm environment. This article explains how to work with ASP.NET Core's built-in distributed cache abstractions and implementations.
[View or download sample code](https://github.com/aspnet/Docs/tree/master/aspnetcore/performance/caching/distributed/sample) ([how to download](xref:tutorials/index#how-to-download-a-sample))
2016-10-29 01:35:15 +08:00
2017-11-18 03:22:43 +08:00
## What is a distributed cache
2016-10-29 01:35:15 +08:00
A distributed cache is shared by multiple app servers (see [Cache Basics](memory.md#caching-basics)). The information in the cache isn't stored in the memory of individual web servers, and the cached data is available to all of the app's servers. This provides several advantages:
2016-10-29 01:35:15 +08:00
1. Cached data is coherent on all web servers. Users don't see different results depending on which web server handles their request
2. Cached data survives web server restarts and deployments. Individual web servers can be removed or added without impacting the cache.
3. The source data store has fewer requests made to it (than with multiple in-memory caches or no cache at all).
> [!NOTE]
> If using a SQL Server Distributed Cache, some of these advantages are only true if a separate database instance is used for the cache than for the app's source data.
Like any cache, a distributed cache can dramatically improve an app's responsiveness, since typically data can be retrieved from the cache much faster than from a relational database (or web service).
Cache configuration is implementation specific. This article describes how to configure both Redis and SQL Server distributed caches. Regardless of which implementation is selected, the app interacts with the cache using a common `IDistributedCache` interface.
2016-10-29 01:35:15 +08:00
## The IDistributedCache Interface
The `IDistributedCache` interface includes synchronous and asynchronous methods. The interface allows items to be added, retrieved, and removed from the distributed cache implementation. The `IDistributedCache` interface includes the following methods:
2016-10-29 01:35:15 +08:00
**Get, GetAsync**
Takes a string key and retrieves a cached item as a `byte[]` if found in the cache.
**Set, SetAsync**
Adds an item (as `byte[]`) to the cache using a string key.
**Refresh, RefreshAsync**
Refreshes an item in the cache based on its key, resetting its sliding expiration timeout (if any).
**Remove, RemoveAsync**
Removes a cache entry based on its key.
To use the `IDistributedCache` interface:
2016-10-29 01:35:15 +08:00
1. Add the required NuGet packages to your project file.
2016-10-29 01:35:15 +08:00
2. Configure the specific implementation of `IDistributedCache` in your `Startup` class's `ConfigureServices` method, and add it to the container there.
2016-10-29 01:35:15 +08:00
3. From the app's [Middleware](xref:fundamentals/middleware/index) or MVC controller classes, request an instance of `IDistributedCache` from the constructor. The instance will be provided by [Dependency Injection](../../fundamentals/dependency-injection.md) (DI).
2016-10-29 01:35:15 +08:00
> [!NOTE]
> There's no need to use a Singleton or Scoped lifetime for `IDistributedCache` instances (at least for the built-in implementations). You can also create an instance wherever you might need one (instead of using [Dependency Injection](../../fundamentals/dependency-injection.md)), but this can make your code harder to test, and violates the [Explicit Dependencies Principle](http://deviq.com/explicit-dependencies-principle/).
2016-10-29 01:35:15 +08:00
The following example shows how to use an instance of `IDistributedCache` in a simple middleware component:
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
[!code-csharp[](distributed/sample/src/DistCacheSample/StartTimeHeader.cs)]
2016-10-29 01:35:15 +08:00
In the code above, the cached value is read, but never written. In this sample, the value is only set when a server starts up, and doesn't change. In a multi-server scenario, the most recent server to start will overwrite any previous values that were set by other servers. The `Get` and `Set` methods use the `byte[]` type. Therefore, the string value must be converted using `Encoding.UTF8.GetString` (for `Get`) and `Encoding.UTF8.GetBytes` (for `Set`).
The following code from *Startup.cs* shows the value being set:
2018-06-05 01:03:05 +08:00
[!code-csharp[](distributed/sample/src/DistCacheSample/Startup.cs?name=snippet1)]
2016-10-29 01:35:15 +08:00
> [!NOTE]
> Since `IDistributedCache` is configured in the `ConfigureServices` method, it's available to the `Configure` method as a parameter. Adding it as a parameter will allow the configured instance to be provided through DI.
2016-10-29 01:35:15 +08:00
2017-11-18 03:22:43 +08:00
## Using a Redis distributed cache
2016-10-29 01:35:15 +08:00
[Redis](https://redis.io/) is an open source in-memory data store, which is often used as a distributed cache. You can use it locally, and you can configure an [Azure Redis Cache](https://azure.microsoft.com/services/cache/) for your Azure-hosted ASP.NET Core apps. Your ASP.NET Core app configures the cache implementation using a `RedisDistributedCache` instance.
2016-10-29 01:35:15 +08:00
You configure the Redis implementation in `ConfigureServices` and access it in your app code by requesting an instance of `IDistributedCache` (see the code above).
2016-10-29 01:35:15 +08:00
In the sample code, a `RedisCache` implementation is used when the server is configured for a `Staging` environment. Thus the `ConfigureStagingServices` method configures the `RedisCache`:
2018-06-05 01:03:05 +08:00
[!code-csharp[](distributed/sample/src/DistCacheSample/Startup.cs?name=snippet2)]
2016-10-29 01:35:15 +08:00
> [!NOTE]
> To install Redis on your local machine, install the chocolatey package [https://chocolatey.org/packages/redis-64/](https://chocolatey.org/packages/redis-64/) and run `redis-server` from a command prompt.
2016-10-29 01:35:15 +08:00
2017-11-18 03:22:43 +08:00
## Using a SQL Server distributed cache
2016-10-29 01:35:15 +08:00
The SqlServerCache implementation allows the distributed cache to use a SQL Server database as its backing store. To create SQL Server table you can use sql-cache tool, the tool creates a table with the name and schema you specify.
2018-06-05 01:03:05 +08:00
::: moniker range="< aspnetcore-2.1"
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
Add `SqlConfig.Tools` to the `<ItemGroup>` element of the project file and run `dotnet restore`.
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
```xml
<ItemGroup>
<DotNetCliToolReference Include="Microsoft.Extensions.Caching.SqlConfig.Tools"
Version="2.0.2" />
</ItemGroup>
```
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
::: moniker-end
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
Test SqlConfig.Tools by running the following command:
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
```console
dotnet sql-cache create --help
```
SqlConfig.Tools displays usage, options, and command help.
Create a table in SQL Server by running the `sql-cache create` command :
```console
dotnet sql-cache create "Data Source=(localdb)\v11.0;Initial Catalog=DistCache;Integrated Security=True;" dbo TestCache
info: Microsoft.Extensions.Caching.SqlConfig.Tools.Program[0]
Table and index were created successfully.
```
2016-10-29 01:35:15 +08:00
2017-02-15 05:34:42 +08:00
The created table has the following schema:
2016-10-29 01:35:15 +08:00
![SqlServer Cache Table](distributed/_static/SqlServerCacheTable.png)
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
Like all cache implementations, your app should get and set cache values using an instance of `IDistributedCache`, not a `SqlServerCache`. The sample implements `SqlServerCache` in the Production environment (so it's configured in `ConfigureProductionServices`).
2016-10-29 01:35:15 +08:00
2018-06-05 01:03:05 +08:00
[!code-csharp[](distributed/sample/src/DistCacheSample/Startup.cs?name=snippet3)]
2016-10-29 01:35:15 +08:00
> [!NOTE]
> The `ConnectionString` (and optionally, `SchemaName` and `TableName`) should typically be stored outside of source control (such as UserSecrets), as they may contain credentials.
## Recommendations
2018-05-12 04:51:27 +08:00
When deciding which implementation of `IDistributedCache` is right for your app, choose between Redis and SQL Server based on your existing infrastructure and environment, your performance requirements, and your team's experience. If your team is more comfortable working with Redis, it's an excellent choice. If your team prefers SQL Server, you can be confident in that implementation as well. Note that a traditional caching solution stores data in-memory which allows for fast retrieval of data. You should store commonly used data in a cache and store the entire data in a backend persistent store such as SQL Server or Azure Storage. Redis Cache is a caching solution which gives you high throughput and low latency as compared to SQL Cache.
2016-10-29 01:35:15 +08:00
2017-11-18 03:22:43 +08:00
## Additional resources
2016-10-29 01:35:15 +08:00
2017-07-11 02:20:50 +08:00
* [Redis Cache on Azure](https://azure.microsoft.com/documentation/services/redis-cache/)
* [SQL Database on Azure](https://azure.microsoft.com/documentation/services/sql-database/)
* [Cache in-memory](xref:performance/caching/memory)
2017-11-18 03:22:43 +08:00
* [Detect changes with change tokens](xref:fundamentals/primitives/change-tokens)
* [Response caching](xref:performance/caching/response)
* [Response Caching Middleware](xref:performance/caching/middleware)
* [Cache Tag Helper](xref:mvc/views/tag-helpers/builtin-th/cache-tag-helper)
* [Distributed Cache Tag Helper](xref:mvc/views/tag-helpers/builtin-th/distributed-cache-tag-helper)