Кеширование в WCF? - PullRequest
       28

Кеширование в WCF?

42 голосов
/ 28 мая 2009

Я строю сервис WCF. Мне нужно хранить справочные данные в кеше, который я буду искать каждый раз, когда получаю входные данные от метода ... Как правильно это сделать? Я также хотел бы определить политику истечения срока действия для кэша, которая сделает его недействительным через определенный промежуток времени.

Ответы [ 9 ]

30 голосов
/ 10 августа 2011

Если вы используете .NET 4, рекомендуется использовать MemoryCache

28 голосов
/ 26 ноября 2011

Любое решение для кэширования должно решить две основные проблемы

1) Хранение элементов кэша и поиск

2) Аннулирование кэша

Поскольку Http-кэширование хорошо известно, я не буду его подробно объяснять. Вы можете использовать атрибут совместимости asp отдельно с некоторыми веб-конфигурациями, где вы получите кеширование по очарованию.

[AspNetCacheProfile("MyProfile")]
        public Customer GetName(string id)
        {
             // ...
        }

И веб-конфигурация похожа на

<system.serviceModel>
        <serviceHostingEnvironment aspNetCompatibilityEnabled="true" />    
</system.serviceModel>
<system.web>
   <caching>
      <outputCacheSettings>
         <outputCacheProfiles>
            <add name=" MyProfile" duration="600" varyByParam="none" sqlDependency="MyTestDatabase:MyTable"/>
         </outputCacheProfiles>
      </outputCacheSettings>
   </caching>
</system.web>

Но это не подходит для большинства сценариев, особенно если у вас есть большой сложный объект для кэширования. Например, у меня была ситуация, когда я хотел кэшировать изображение, сгенерированное системой (вывод контракта операции - это изображение, сгенерированное системой, которое зависит от ввода). В таком случае вы должны реализовать свой собственный кеш. Я использовал блоки кэширования корпоративной библиотеки Microsoft, которые отвечали всем моим требованиям к хранилищу. Тем не менее, вам все еще нужно выполнить подключение, чтобы интегрировать блок кэширования корпоративной библиотеки Microsoft с вашей службой WCF. Сначала необходимо перехватить канал связи WCF для реализации кэша. Подробное обсуждение того, как перехватить канал связи WCF, можно найти по адресу http://msdn.microsoft.com/en-us/magazine/cc163302.aspx. Вот как вы делаете сантехнику для кеширования WCF

Basic Plumbing Architecture

Шаг 0 Допустим, у вас есть контракт на выполнение операций следующим образом, и вы хотите кэшировать возврат элемента этим методом.

[OperationContract]
MyCompositeClass Rotate(int angle)

Шаг 1 Сначала вы должны зарегистрировать свой пользовательский кэшер в конвейере WCF. Для этого я собираюсь использовать атрибут, чтобы я мог красиво украсить свой вызов WCF в соответствии с принципами программирования аспекта.

using System;
using System.ServiceModel.Description;
using System.ServiceModel.Channels;
using System.ServiceModel.Dispatcher;
using System.Reflection;

    [AttributeUsage(AttributeTargets.Method)]
    public class MyCacheRegister : Attribute, IOperationBehavior
    {
        ConstructorInfo _chacherImplementation;
        public ImageCache(Type provider)
        {
            if (provider == null)
            {
                throw new ArgumentNullException("Provider can't be null");
            }
            else if (provider.IsAssignableFrom(typeof(IOperationInvoker)))
            {
                throw new ArgumentException("The type " + provider.AssemblyQualifiedName + " does not implements the interface " + typeof(IOperationInvoker).AssemblyQualifiedName);
            }
            else
            {
                try
                {
                    Type[] constructorSignatureTypes = new Type[1];
                    constructorSignatureTypes[0] = typeof(IOperationInvoker);
                    _chacherImplementation = provider.GetConstructor(constructorSignatureTypes);

                }
                catch
                {
                    throw new ArgumentException("There is no constructor in " + provider.AssemblyQualifiedName + " that accept " + typeof(IOperationInvoker).AssemblyQualifiedName + " as a parameter");
                }

            }


        }

        public void AddBindingParameters(OperationDescription operationDescription, BindingParameterCollection bindingParameters)
        {
            return;
        }

        public void ApplyClientBehavior(OperationDescription operationDescription, ClientOperation clientOperation)
        {
            return;
        }

        /// <summary>
        /// Decorate the method call with the cacher
        /// </summary>
        public void ApplyDispatchBehavior(OperationDescription operationDescription, DispatchOperation dispatchOperation)
        {
            //decorator pattern, decorate with a  cacher
            object[] constructorParam = new object[1];
            constructorParam[0] = dispatchOperation.Invoker;
            dispatchOperation.Invoker = (IOperationInvoker)_chacherImplementation.Invoke(constructorParam);
        }

        public void Validate(OperationDescription operationDescription)
        {
            return;
        }
    }

Шаг 2

Затем вам нужно реализовать точку, в которой будет извлечен объект кэша.

using System;
using System.ServiceModel.Dispatcher;
using Microsoft.Practices.EnterpriseLibrary.Caching;
using Microsoft.Practices.EnterpriseLibrary.Common;
using System.IO;

    class RotateCacher : IOperationInvoker
    {

        private IOperationInvoker _innerOperationInvoker;
        public RotateImageCacher(IOperationInvoker innerInvoker)
        {
            _innerOperationInvoker = innerInvoker;
        }
        public object[] AllocateInputs()
        {
            Object[] result = _innerOperationInvoker.AllocateInputs();
            return result;
        }

        public object Invoke(object instance, object[] inputs, out object[] outputs)
        {
            object result=null;

///TODO: You will have more object in the input if you have more ///parameters in your method

            string angle = inputs[1].ToString();

            ///TODO: create a unique key from the inputs
            string key = angle;

            string provider = System.Configuration.ConfigurationManager.AppSettings["CacheProviderName"];
            ///Important Provider will be DiskCache or MemoryCache for the moment
provider =”DiskCache”;
///TODO: call enterprise library cache manager, You can have your own 
/// custom cache like Hashtable

    ICacheManager manager = CacheFactory.GetCacheManager(provider);

            if (manager.Contains(key))
            {

                result =(MyCompositeClass) manager[key];

            }
            else
            {
                result =(MyCompositeClass) _innerOperationInvoker.Invoke(instance, inputs, out outputs);
                manager.Add(key, result);
            }
            return result;
        }

        public IAsyncResult InvokeBegin(object instance, object[] inputs, AsyncCallback callback, object state)
        {
            IAsyncResult result = _innerOperationInvoker.InvokeBegin(instance, inputs, callback, state);
            return result;
        }

        public object InvokeEnd(object instance, out object[] outputs, IAsyncResult asyncResult)
        {
            object result = _innerOperationInvoker.InvokeEnd(instance, out outputs, asyncResult);
            return result;
        }

        public bool IsSynchronous
        {
            get { return _innerOperationInvoker.IsSynchronous; }
        }
    }

Шаг 3

Наконец добавьте свой атрибут выше вашего сервисного вызова

[OperationContract]
[MyCacheRegister(typeof(RotateCacher)]
MyCompositeClass Rotate(int angle)

Конфигурация блока кэширования корпоративной библиотеки выходит за рамки этого ответа. Вы можете использовать следующую ссылку, чтобы узнать это. Преимущество корпоративной библиотеки состоит в том, что вы получаете готовые способы расширения политики кэширования. Он имеет встроенные способы для истечения срока действия кэша и хранения. Вы также можете написать свои собственные политики истечения срока хранения и хранения. http://www.codeproject.com/KB/web-cache/CachingApplicationBlock.aspx

И, наконец, для того, чтобы кеширование корпоративной библиотеки работало, вам необходимо добавить следующие сведения о конфигурации. Вы также должны добавить соответствующие dll к вашей ссылке проекта.

<configSections>
    <section name="cachingConfiguration" type="Microsoft.Practices.EnterpriseLibrary.Caching.Configuration.CacheManagerSettings, Microsoft.Practices.EnterpriseLibrary.Caching, Version=5.0.414.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35" requirePermission="true" />
  </configSections>


  <cachingConfiguration defaultCacheManager="Cache Manager">
    <cacheManagers>
      <add name="MemoryCache" type="Microsoft.Practices.EnterpriseLibrary.Caching.CacheManager, Microsoft.Practices.EnterpriseLibrary.Caching, Version=5.0.414.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35"
        expirationPollFrequencyInSeconds="60" maximumElementsInCacheBeforeScavenging="1000"
        numberToRemoveWhenScavenging="10" backingStoreName="NullBackingStore" />
      <add name="DiskCache" type="Microsoft.Practices.EnterpriseLibrary.Caching.CacheManager, Microsoft.Practices.EnterpriseLibrary.Caching, Version=5.0.414.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35"
        expirationPollFrequencyInSeconds="60" maximumElementsInCacheBeforeScavenging="1000"
        numberToRemoveWhenScavenging="10" backingStoreName="IsolatedStorageCacheStore" />
    </cacheManagers>
    <backingStores>
      <add type="Microsoft.Practices.EnterpriseLibrary.Caching.BackingStoreImplementations.NullBackingStore, Microsoft.Practices.EnterpriseLibrary.Caching, Version=5.0.414.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35"
        name="NullBackingStore" />
      <add name="IsolatedStorageCacheStore" type="Microsoft.Practices.EnterpriseLibrary.Caching.BackingStoreImplementations.IsolatedStorageBackingStore, Microsoft.Practices.EnterpriseLibrary.Caching, Version=5.0.414.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35"
        encryptionProviderName="" partitionName="MyCachePartition" />
    </backingStores>
  </cachingConfiguration>
1 голос
/ 08 октября 2013

Если вы собираетесь масштабировать до нескольких серверов в системе с балансировкой нагрузки и без сохранения состояния, вам нужно спроектировать использование распределенного кэша . Основные вещи, которые нужно сделать здесь:

  1. Использовать как локальный, так и распределенный кеш. Только поставить сессию или короткую живые вещи в распределенном кеше, другие вещи кешируются локально.

  2. Установите соответствующие таймауты для элементов. Это будет варьироваться в зависимости от тип информации и насколько близко к источнику она должна быть.

  3. Удалить содержимое из кэша, когда вы знаете, что это будет недержание мочи (например, обновления, удаления и т. д.).

  4. Позаботьтесь о том, чтобы создавать уникальные ключи кеша. Построить модель тип информации, которую вы планируете кешировать и использовать в качестве шаблона для строительные ключи.

1 голос
/ 09 мая 2011
1 голос
/ 28 мая 2009

Вы можете взглянуть на Скорость . Это распределенная инфраструктура кэширования Microsoft в памяти . Но это может быть слишком бета ...

0 голосов
/ 16 июня 2011

Вместо того, чтобы время от времени истекать данные кеша, вы можете просто сделать недействительным кеш всякий раз, когда изменяются базовые данные, которые вы кэшируете.

См. Этот пример в информации Q http://www.infoq.com/news/2011/04/Attribute-Caching

[Cache.Cacheable("UserTransactionCache")]
public DataTable GetAllTransactionsForUser(int userId)
{
    return new DataProvider().GetAllTransactionsForUser(userId);
}

[Cache.TriggerInvalidation("UserTransactionCache")]
public void DeleteAllTransactionsForUser(int userId)
{
 ...
}
0 голосов
/ 28 мая 2009

Начальный комплект WCF REST имеет кеширование, вот статья об его использовании ... с примером кода.

http://weblogs.asp.net/gsusx/archive/2008/10/29/adding-caching-to-wcf-restful-services-using-the-rest-starter-kit.aspx

0 голосов
/ 28 мая 2009

Есть много способов сделать это. Один довольно простой способ - разместить объект System.Web.Cache самостоятельно и использовать его для хранения справочных данных. Вот хороший пример этого: http://kjellsj.blogspot.com/2007/11/wcf-caching-claims-using.html

0 голосов
/ 28 мая 2009

Вы можете использовать System.Web.Cache (даже если вы не в веб-контексте), и это то, что я бы сделал. По сути, это большая хеш-таблица в памяти с некоторыми тонкостями для устаревшего содержимого.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...