1

私は最近、新しい Outlook/Office 365 Rest API の使用を開始し、認証(新しい v2.0 アプリ モデル)、スコープOutlookServicesClientの使用、より優れた(linq-)queriesの作成に関して大きな助けを得ました。

残念ながら、一部のユーザーからメモリの問題が報告され、次のような結果になりましたSystem.OutOfMemoryException

    Exception of type 'System.OutOfMemoryException' was thrown. :    at Microsoft.OData.Core.Json.JsonReader.ReadInput()
       at Microsoft.OData.Core.Json.JsonReader.ParseStringPrimitiveValue(Boolean& hasLeadingBackslash)
       at Microsoft.OData.Core.Json.JsonReader.ParseValue()
       at Microsoft.OData.Core.Json.JsonReader.Read()
       at Microsoft.OData.Core.Json.BufferingJsonReader.ReadInternal()
       at Microsoft.OData.Core.Json.BufferingJsonReader.ReadNextAndCheckForInStreamError()
       at Microsoft.OData.Core.Json.BufferingJsonReader.ReadInternal()
       at Microsoft.OData.Core.Json.BufferingJsonReader.Read()
       at Microsoft.OData.Core.JsonLight.ODataJsonLightDeserializer.ParseProperty(DuplicatePropertyNamesChecker duplicatePropertyNamesChecker, Func`2 readPropertyAnnotationValue, String& parsedPropertyName)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightDeserializer.ProcessProperty(DuplicatePropertyNamesChecker duplicatePropertyNamesChecker, Func`2 readPropertyAnnotationValue, Action`2 handleProperty)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightPropertyAndValueDeserializer.ReadComplexValue(IEdmComplexTypeReference complexValueTypeReference, String payloadTypeName, SerializationTypeNameAnnotation serializationTypeNameAnnotation, DuplicatePropertyNamesChecker duplicatePropertyNamesChecker)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightPropertyAndValueDeserializer.ReadNonEntityValueImplementation(String payloadTypeName, IEdmTypeReference expectedTypeReference, DuplicatePropertyNamesChecker duplicatePropertyNamesChecker, CollectionWithoutExpectedTypeValidator collectionValidator, Boolean validateNullValue, Boolean isTopLevelPropertyValue, Boolean insideComplexValue, String propertyName, Nullable`1 isDynamicProperty)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightEntryAndFeedDeserializer.ReadEntryDataProperty(IODataJsonLightReaderEntryState entryState, IEdmProperty edmProperty, String propertyTypeName)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightEntryAndFeedDeserializer.ReadEntryPropertyWithValue(IODataJsonLightReaderEntryState entryState, String propertyName)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightEntryAndFeedDeserializer.<>c__DisplayClass2.<ReadEntryContent>b__0(PropertyParsingResult propertyParsingResult, String propertyName)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightDeserializer.ProcessProperty(DuplicatePropertyNamesChecker duplicatePropertyNamesChecker, Func`2 readPropertyAnnotationValue, Action`2 handleProperty)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightEntryAndFeedDeserializer.ReadEntryContent(IODataJsonLightReaderEntryState entryState)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightReader.ReadEntryStart(DuplicatePropertyNamesChecker duplicatePropertyNamesChecker, SelectedPropertiesNode selectedProperties)
       at Microsoft.OData.Core.JsonLight.ODataJsonLightReader.ReadAtEntryEndImplementationSynchronously()
       at Microsoft.OData.Core.JsonLight.ODataJsonLightReader.ReadAtEntryEndImplementation()
       at Microsoft.OData.Core.ODataReaderCore.ReadImplementation()
       at Microsoft.OData.Core.ODataReaderCore.ReadSynchronously()
       at Microsoft.OData.Core.ODataReaderCore.InterceptException[T](Func`1 action)
       at Microsoft.OData.Core.ODataReaderCore.Read()
       at Microsoft.OData.Client.Materialization.ODataReaderWrapper.Read()
       at Microsoft.OData.Client.Materialization.FeedAndEntryMaterializerAdapter.TryRead()
       at Microsoft.OData.Client.Materialization.FeedAndEntryMaterializerAdapter.TryStartReadFeedOrEntry()
       at Microsoft.OData.Client.Materialization.FeedAndEntryMaterializerAdapter.TryReadEntry(MaterializerEntry& entry)
       at Microsoft.OData.Client.Materialization.FeedAndEntryMaterializerAdapter.<LazyReadEntries>d__0.MoveNext()
       at Microsoft.OData.Client.Materialization.FeedAndEntryMaterializerAdapter.Read()
       at Microsoft.OData.Client.Materialization.ODataReaderEntityMaterializer.ReadNextFeedOrEntry()
       at Microsoft.OData.Client.Materialization.ODataEntityMaterializer.ReadImplementation()
       at Microsoft.OData.Client.MaterializeAtom.MoveNextInternal()
       at Microsoft.OData.Client.MaterializeAtom.MoveNext()
       at System.Linq.Enumerable.<CastIterator>d__94`1.MoveNext()
       at System.Collections.Generic.List`1..ctor(IEnumerable`1 collection)
       at System.Linq.Enumerable.ToList[TSource](IEnumerable`1 source)
       at Microsoft.OData.ProxyExtensions.PagedCollection`2..ctor(DataServiceContextWrapper context, QueryOperationResponse`1 qor)
       at Microsoft.OData.ProxyExtensions.DataServiceContextWrapper.<>c__DisplayClass38`2.<ExecuteAsync>b__36(IAsyncResult r)
       at System.Threading.Tasks.TaskFactory`1.FromAsyncCoreLogic(IAsyncResult iar, Func`2 endFunction, Action`1 endAction, Task`1 promise, Boolean requiresSynchronization)
    --- End of stack trace from previous location where exception was thrown ---
       at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
       at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
       at Microsoft.OData.ProxyExtensions.DataServiceContextWrapper.<ExecuteAsync>d__3a`2.MoveNext()
    --- End of stack trace from previous location where exception was thrown ---
       at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
       at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
       at System.Runtime.CompilerServices.TaskAwaiter.ValidateEnd(Task task)
       at Shared.Data.Office365.Office365Api.<GetEmailsReceived>d__23.MoveNext()

私のクエリ(日付に受信した電子メールの数を取得するため)は次のようになります。

var dtStart = date.Date.ToUniversalTime();
var dtEnd = date.Date.AddDays(1).ToUniversalTime();

var mailResults = await _client.Me.Messages
    .OrderByDescending(m => m.ReceivedDateTime)
    .Where(m => m.ReceivedDateTime.Value >= dtStart && m.ReceivedDateTime.Value <= dtEnd)
    //todo: filter if not in Junk Email and Deleted Folder (maybe with ParentFolderId)
    .Take(500) // todo: add paging to (and maybe make more efficient?)
    .Select(m => new DisplayEmail(m)).ExecuteAsync();

// todo: add paging to (and maybe make more efficient?)
foreach (var message in mailResults.CurrentPage)
{
    emails.Add(message);
}

ソースを破棄する方法が見つかりませんでした。また、ローカル変数しかないため、それが必要だとも思いません。emails-listの結果はデー​​タベースに保存されます。ツールのメモリ使用量は、この機能を追加する前は安定していましたが、現在は 1 時間あたり約 10Mb 増加しています (メールの数によって異なります)。

どんな提案でも大歓迎です!

4

1 に答える 1

2

ここでいくつかのことを試してみてください:

  • ページ サイズを小さくします。ここでは一度に 500 を実行しています ( .Take(500))。それを減らしてみてください。
  • 関心のあるプロパティのみをリクエストします。変数.Selectの特定のプロパティを参照するように変更する場合、APIは API リクエストでクエリ パラメータを使用して、返されるものを制限する必要があります。例については、このチュートリアルを参照してください。m$select
于 2016-02-04T14:50:35.600 に答える