Code:
/ Net / Net / 3.5.50727.3053 / DEVDIV / depot / DevDiv / releases / Orcas / SP / ndp / fx / src / DataEntity / System / Data / Common / Utils / Memoizer.cs / 2 / Memoizer.cs
//---------------------------------------------------------------------- //// Copyright (c) Microsoft Corporation. All rights reserved. // // // @owner [....], [....] //--------------------------------------------------------------------- using System.Collections.Generic; using System.Threading; using System.Diagnostics; namespace System.Data.Common.Utils { ////// Remembers the result of evaluating an expensive function so that subsequent /// evaluations are faster. Thread-safe. /// ///Type of the argument to the function. ///Type of the function result. internal sealed class Memoizer{ private readonly Func _function; private readonly Dictionary _resultCache; private readonly ReaderWriterLockSlim _lock; /// /// Constructs /// /// Required. Function whose values are being cached. /// Optional. Comparer used to determine if two functions arguments /// are the same. internal Memoizer(Funcfunction, IEqualityComparer argComparer) { EntityUtil.CheckArgumentNull(function, "function"); _function = function; _resultCache = new Dictionary (argComparer); _lock = new ReaderWriterLockSlim(); } /// /// Evaluates the wrapped function for the given argument. If the function has already /// been evaluated for the given argument, returns cached value. Otherwise, the value /// is computed and returned. /// /// Function argument. ///Function result. internal TResult Evaluate(TArg arg) { Result result; bool hasResult; // check to see if a result has already been computed _lock.EnterReadLock(); try { hasResult = _resultCache.TryGetValue(arg, out result); } finally { _lock.ExitReadLock(); } if (!hasResult) { // compute the new value _lock.EnterWriteLock(); try { // see if the value has been computed in the interim if (!_resultCache.TryGetValue(arg, out result)) { result = new Result(() => _function(arg)); _resultCache.Add(arg, result); } } finally { _lock.ExitWriteLock(); } } // note: you need to release the global cache lock before (potentially) acquiring // a result lock in result.GetValue() return result.GetValue(); } ////// Encapsulates a 'deferred' result. The result is constructed with a delegate (must not /// be null) and when the user requests a value the delegate is invoked and stored. /// private class Result { private TResult _value; private Func_delegate; internal Result(Func createValueDelegate) { Debug.Assert(null != createValueDelegate, "delegate must be given"); _delegate = createValueDelegate; } internal TResult GetValue() { if (null == _delegate) { // if the delegate has been cleared, it means we have already computed the value return _value; } // lock the entry while computing the value so that two threads // don't simultaneously do the work lock (this) { if (null == _delegate) { // between our initial check and our acquisition of the lock, some other // thread may have computed the value return _value; } _value = _delegate(); // ensure _delegate (and its closure) is garbage collected, and set to null // to indicate that the value has been computed _delegate = null; return _value; } } } } } // File provided for Reference Use Only by Microsoft Corporation (c) 2007. //---------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // // @owner [....], [....] //--------------------------------------------------------------------- using System.Collections.Generic; using System.Threading; using System.Diagnostics; namespace System.Data.Common.Utils { ////// Remembers the result of evaluating an expensive function so that subsequent /// evaluations are faster. Thread-safe. /// ///Type of the argument to the function. ///Type of the function result. internal sealed class Memoizer{ private readonly Func _function; private readonly Dictionary _resultCache; private readonly ReaderWriterLockSlim _lock; /// /// Constructs /// /// Required. Function whose values are being cached. /// Optional. Comparer used to determine if two functions arguments /// are the same. internal Memoizer(Funcfunction, IEqualityComparer argComparer) { EntityUtil.CheckArgumentNull(function, "function"); _function = function; _resultCache = new Dictionary (argComparer); _lock = new ReaderWriterLockSlim(); } /// /// Evaluates the wrapped function for the given argument. If the function has already /// been evaluated for the given argument, returns cached value. Otherwise, the value /// is computed and returned. /// /// Function argument. ///Function result. internal TResult Evaluate(TArg arg) { Result result; bool hasResult; // check to see if a result has already been computed _lock.EnterReadLock(); try { hasResult = _resultCache.TryGetValue(arg, out result); } finally { _lock.ExitReadLock(); } if (!hasResult) { // compute the new value _lock.EnterWriteLock(); try { // see if the value has been computed in the interim if (!_resultCache.TryGetValue(arg, out result)) { result = new Result(() => _function(arg)); _resultCache.Add(arg, result); } } finally { _lock.ExitWriteLock(); } } // note: you need to release the global cache lock before (potentially) acquiring // a result lock in result.GetValue() return result.GetValue(); } ////// Encapsulates a 'deferred' result. The result is constructed with a delegate (must not /// be null) and when the user requests a value the delegate is invoked and stored. /// private class Result { private TResult _value; private Func_delegate; internal Result(Func createValueDelegate) { Debug.Assert(null != createValueDelegate, "delegate must be given"); _delegate = createValueDelegate; } internal TResult GetValue() { if (null == _delegate) { // if the delegate has been cleared, it means we have already computed the value return _value; } // lock the entry while computing the value so that two threads // don't simultaneously do the work lock (this) { if (null == _delegate) { // between our initial check and our acquisition of the lock, some other // thread may have computed the value return _value; } _value = _delegate(); // ensure _delegate (and its closure) is garbage collected, and set to null // to indicate that the value has been computed _delegate = null; return _value; } } } } } // File provided for Reference Use Only by Microsoft Corporation (c) 2007.
Link Menu
This book is available now!
Buy at Amazon US or
Buy at Amazon UK
- StaticSiteMapProvider.cs
- XmlWhitespace.cs
- SynchronizationFilter.cs
- AuthorizationRule.cs
- XdrBuilder.cs
- JsonStringDataContract.cs
- EntityCommandCompilationException.cs
- RequestResizeEvent.cs
- SmiTypedGetterSetter.cs
- SystemDropShadowChrome.cs
- PasswordDeriveBytes.cs
- DataColumnMappingCollection.cs
- PropertySourceInfo.cs
- CurrencyWrapper.cs
- Icon.cs
- WebPartCatalogCloseVerb.cs
- TypeBuilder.cs
- RelationshipDetailsCollection.cs
- XomlCompilerParameters.cs
- DataPager.cs
- SafeSecurityHelper.cs
- DecimalKeyFrameCollection.cs
- ConnectionStringSettings.cs
- RequestTimeoutManager.cs
- HtmlInputPassword.cs
- TreeWalker.cs
- HtmlDocument.cs
- DataGridViewRowHeightInfoPushedEventArgs.cs
- NamespaceListProperty.cs
- CompleteWizardStep.cs
- StringComparer.cs
- ActivityBuilder.cs
- DataColumnCollection.cs
- TabletDevice.cs
- ZipIOLocalFileBlock.cs
- CaseStatementSlot.cs
- DateTimeConverter.cs
- ByteStreamGeometryContext.cs
- counter.cs
- _OverlappedAsyncResult.cs
- MultiPropertyDescriptorGridEntry.cs
- externdll.cs
- MediaTimeline.cs
- RemotingSurrogateSelector.cs
- WebScriptEnablingBehavior.cs
- JavaScriptObjectDeserializer.cs
- ResXResourceWriter.cs
- XamlTreeBuilder.cs
- DesignerDataParameter.cs
- ImageClickEventArgs.cs
- RenderCapability.cs
- FontWeights.cs
- RelationshipConstraintValidator.cs
- RangeValidator.cs
- WebPartConnectionsConnectVerb.cs
- VersionUtil.cs
- RoutedEvent.cs
- XamlValidatingReader.cs
- ConstNode.cs
- PanelDesigner.cs
- safelink.cs
- EventlogProvider.cs
- FakeModelItemImpl.cs
- ChangeProcessor.cs
- SqlLiftIndependentRowExpressions.cs
- Menu.cs
- EmptyQuery.cs
- WinFormsSecurity.cs
- ColumnHeaderConverter.cs
- TextureBrush.cs
- MenuStrip.cs
- SafeNativeMethods.cs
- QueryOptionExpression.cs
- TraceListener.cs
- DoubleLinkList.cs
- WebUtil.cs
- FixedPageProcessor.cs
- ActivationServices.cs
- DbCommandDefinition.cs
- QueryOptionExpression.cs
- SystemNetworkInterface.cs
- StateValidator.cs
- CultureTable.cs
- PrintingPermissionAttribute.cs
- ContactManager.cs
- PartEditor.cs
- TransportContext.cs
- ToolStripSystemRenderer.cs
- ContextInformation.cs
- SortFieldComparer.cs
- BulletedListEventArgs.cs
- DtdParser.cs
- xml.cs
- ConfigurationPropertyCollection.cs
- UnsafeMethods.cs
- XmlNodeChangedEventArgs.cs
- InheritanceContextChangedEventManager.cs
- _RequestCacheProtocol.cs
- DbProviderConfigurationHandler.cs
- HttpWebResponse.cs