In Memory Caching ================= By `Steve Smith`_ Caching involves keeping a copy of data in a location that can be accessed more quickly than the source data. ASP.NET 5 has rich support for caching in a variety of ways, including keeping data in memory on the local server, which is referred to as *in memory caching*. Sections: - `Caching Basics`_ - `Configuring In Memory Caching`_ - `Reading and Writing to a Memory Cache`_ - `Cache Dependencies and Callbacks`_ `View or download sample from GitHub `_. .. _caching-basics: Caching Basics -------------- Caching can dramatically improve the performance and scalability of ASP.NET applications, by eliminating unnecessary requests to external data sources for data that changes infrequently. .. note:: Caching in all forms (in-memory or distributed, including session state) involves making a copy of data in order to optimize performance. The copied data should be considered ephemeral - it could disappear at any time. Apps should be written to not depend on cached data, but use it when available. ASP.NET supports several different kinds of caches, the simplest of which is represented by the `IMemoryCache `_ interface, which represents a cache stored in the memory of the local web server. You should always write (and test!) your application such that it can use cached data if it's available, but otherwise will work correctly using the underlying data source. An in-memory cache is stored in the memory of a single server hosting an ASP.NET app. If an app is hosted by multiple servers in a web farm or cloud hosting environment, the servers may have different values in their local in-memory caches. Apps that will be hosted in server farms or on cloud hosting should use a :doc:`distributed cache ` to avoid cache consistency problems. .. tip:: A common use case for caching is data-driven navigation menus, which rarely change but are frequently read for display within an application. Caching results that do not vary often but which are requested frequently can greatly improve performance by reducing round trips to out of process data stores and unnecessary computation. Configuring In Memory Caching ----------------------------- To use an in memory cache in your ASP.NET application, add the following dependencies to your ``project.json`` file: .. literalinclude:: memory/sample/src/CachingSample/project.json :linenos: :lines: 4-11 :emphasize-lines: 4-5 Caching in ASP.NET 5 is a *service* that should be referenced from your application by :doc:`/fundamentals/dependency-injection`. To register the caching service and make it available within your app, add the following line to your ``ConfigureServices`` method in ``Startup``: .. literalinclude:: memory/sample/src/CachingSample/Startup.cs :linenos: :lines: 12-15 :dedent: 8 :emphasize-lines: 3 You utilize caching in your app by requesting an instance of ``IMemoryCache`` in your controller or middleware constructor. In the sample for this article, we are using a simple middleware component to handle requests by returning customized greeting. The constructor is shown here: .. literalinclude:: memory/sample/src/CachingSample/Middleware/GreetingMiddleware.cs :linenos: :lines: 19-28 :dedent: 8 :emphasize-lines: 2,7 Reading and Writing to a Memory Cache ------------------------------------- The middleware's ``Invoke`` method returns the cached data when it's available. There are two methods for accessing cache entries: ``Get`` ``Get`` will return the value if it exists, but otherwise returns ``null``. ``TryGet`` ``TryGet`` will assign the cached value to an ``out`` parameter and return true if the entry exists. Otherwise it returns false. Use the ``Set`` method to write to the cache. ``Set`` accepts the key to use to look up the value, the value to be cached, and a set of ``MemoryCacheEntryOptions``. The ``MemoryCacheEntryOptions`` allow you to specify absolute or sliding time-based cache expiration, caching priority, callbacks, and dependencies. These options are detailed below. The sample code (shown below) uses the ``SetAbsoluteExpiration`` method on ``MemoryCacheEntryOptions`` to cache greetings for one minute. .. literalinclude:: memory/sample/src/CachingSample/Middleware/GreetingMiddleware.cs :linenos: :lines: 30-58 :dedent: 8 :emphasize-lines: 7,10,16-18 In addition to setting an absolute expiration, a sliding expiration can be used to keep frequently requested items in the cache: .. code-block:: c# // keep item in cache as long as it is requested at least // once every 5 minutes new MemoryCacheEntryOptions() .SetSlidingExpiration(TimeSpan.FromMinutes(5)) To avoid having frequently-accessed cache entries growing too stale (because their sliding expiration is constantly reset), you can combine absolute and sliding expirations: .. code-block:: c# // keep item in cache as long as it is requested at least // once every 5 minutes... // but in any case make sure to refresh it every hour new MemoryCacheEntryOptions() .SetSlidingExpiration(TimeSpan.FromMinutes(5)) .SetAbsoluteExpiration(TimeSpan.FromHours(1)) By default, an instance of ``MemoryCache`` will automatically manage the items stored, removing entries when necessary in response to memory pressure in the app. You can influence the way cache entries are managed by setting their `CacheItemPriority `_ when adding the item to the cache. For instance, if you have an item you want to keep in the cache unless you explicitly remove it, you would use the ``NeverRemove`` priority option: .. code-block:: c# // keep item in cache indefinitely unless explicitly removed new MemoryCacheEntryOptions() .SetPriority(CacheItemPriority.NeverRemove)) When you do want to explicitly remove an item from the cache, you can do so easily using the ``Remove`` method: .. code-block:: c# cache.Remove(cacheKey); Cache Dependencies and Callbacks -------------------------------- You can configure cache entries to depend on other cache entries, the file system, or programmatic tokens, evicting the entry in response to changes. You can register a callback, which will run when a cache item is evicted. .. literalinclude:: memory/sample/test/CachingSample.Tests/MemoryCacheTests.cs :linenos: :lines: 22-41 :dedent: 8 :emphasize-lines: 6-11,18 The callback is run on a different thread from the code that removes the item from the cache. .. warning:: If the callback is used to repopulate the cache it is possible other requests for the cache will take place (and find it empty) before the callback completes, possibly resulting in several threads repopulating the cached value. Possible `eviction reasons `_ are: None No reason known. Removed The item was manually removed by a call to ``Remove()`` Replaced The item was overwritten. Expired The item timed out. TokenExpired The token the item depended upon fired an event. Capacity The item was removed as part of the cache's memory management process. You can specify that one or more cache entries depend on a ``CancellationTokenSource`` by adding the expiration token to the ``MemoryCacheEntryOptions`` object. When a cached item is invalidated, call ``Cancel`` on the token, which will expire all of the associated cache entries (with a reason of ``TokenExpired``). The following unit test demonstrates this: .. literalinclude:: memory/sample/test/CachingSample.Tests/MemoryCacheTests.cs :linenos: :lines: 43-64 :dedent: 8 :emphasize-lines: 7,16,21 Using a ``CancellationTokenSource`` allows multiple cache entries to all be expired without the need to create a dependency between cache entries themselves (in which case, you must ensure that the source cache entry exists before it is used as a dependency for other entries). Use a cache entry link, ``IEntryLink`` to specify that more than one cache entry is linked to the same cancellation token and/or time-based expiration. This approach ensures that subordinate cache entries expire at the same time as related entries. .. literalinclude:: memory/sample/test/CachingSample.Tests/MemoryCacheTests.cs :linenos: :lines: 66-94 :dedent: 8 :emphasize-lines: 5,13,22-23 .. note:: When one cache entry is linked to another, it copies that entry's expiration token and time-based expiration settings, if any. It is not expired in response to manual removal or updating of the linked entry. Other Resources --------------- - :doc:`distributed`