在视图中获取InvalidOperationException.Collection已被修改错误

问题描述

我有一个.Net核心Web应用程序。我的代码可以在调试模式下完美运行,在服务器上,我看不到任何可见的错误。但是此错误反复记录在我的记录器中。

system.invalidOperationException: Collection was modified; enumeration operation may not execute.
 at System.Collections.Generic.List`1.Enumerator.MoveNextRare()
 at System.Linq.Enumerable.TryGetFirst[TSource](IEnumerable`1 source,Func`2 predicate,Boolean& found)
 at System.Linq.Enumerable.First[TSource](IEnumerable`1 source,Func`2 predicate)
 at AspNetCore.Views_Home__ItemsList.ExecuteAsync() in [project path]\Views\Home\_ItemsList.cshtml:line 56

我不知道为什么会发生,我不会在视图中更改列表。

IndexDto

public class IndexDto
{
    public List<SliderListDto> Sliders { get; set; }
    public List<ItemDto> Featured { get; set; }
    public ResourceList<ItemDto> Items { get; set; }        
}

资源列表

public class ResourceList<T>
{
    public ResourceList() { }
    public ResourceList(List<T> items,bool hasMore = false)
    {
        Items = items;
        HasMore = hasMore;
    }
    public List<T> Items { get; set; }
    public bool HasMore { get; set; }
    public List<Link> Links { get; set; } = new List<Link>();
}

Index.cshtml

@model IndexDto
<div class="home">
    @await Html.PartialAsync("_Sliders",Model.Sliders)
    @await Html.PartialAsync("_Featured",Model.Featured)
    @await Html.PartialAsync("_ItemsList",Model.Items)
</div>

_ItemsList.cshtml

@if (Model.Items.Any())
{
    @*some html here*@
    @foreach (var item in Model.Items)
    {
        @*some html here*@
        @if (item.IsOnline)
        {
            @*some html here*@
        }
        @*some html here*@
        @if (item.OnlineUsersCount > 0)
        {
            @*some html here*@
        }           
    }
    
    @if (Model.HasMore)
    {
        var nextLink = Model.Links.First(l => l.Rel == "next"); //This is where the error happens

        if (nextLink != null)
        {
            @*some html here*@
        }
    }
    @*some html here*@
}

我唯一在链接列表上进行修改的时间是在控制器中的此方法中:

public class HomeController : Controller
{

    public async Task<IActionResult> Index()
    {
        var model = await _queryManager.GeMainPage();
        SetResourceLinks(model);
        return View(model);
    }

    private void SetResourceLinks(IndexDto result)
    {
        if (result.Items.HasMore)
        {
            result.Items.Links.Add(new Link(
                "next",Url.Action(nameof(ItemsController.GetItems),"Items",new { pageNumber = 2},Request.Scheme),HttpMethods.Get));
        }
    }
}

我正在此应用上使用负载平衡。可以与之相关吗?

更新

我在_queryManager.GeMainPage()方法中找到了问题的根源。

public async Task<heyatIndexDto> GetMainPage()
{
    var model = await _redisCache.GetorCreateAsync(CacheKeys.IndexModel,entry =>
    {
        entry.AbsoluteExpirationRelativetoNow = TimeSpan.FromSeconds(5);
        return GetMainPageFromDatabase();
    });
    return model;
}

private async Task<IndexDto> GetMainPageFromDatabase()
{
    //....
}

我将数据作为对象存储在Redis中,当我检索它时,它并没有创建新对象并将链接添加到先前的数组中。我更改了将数据存储为字符串然后反序列化的方法问题解决了。

public async Task<heyatIndexDto> GetheyatMainPage()
{
    var serializedData = await _cache.GetorCreateAsync(CacheKeys.heyatIndexModel,entry =>
    {
        entry.AbsoluteExpirationRelativetoNow = TimeSpan.FromSeconds(5);
        return GetheyatMainPageFromDatabase();
    });
    var model = JsonSerializer.Deserialize<heyatIndexDto>(serializedData);
    return model;
}

private async Task<string> GetheyatMainPageFromDatabase()
{
    //....
}

解决方法

我在项目的另一层发现了问题的根源。

在_queryManager.GeMainPage()方法中,我将数据作为对象存储在Redis中,当我检索它时,它并没有创建新对象并将链接添加到先前的数组。我更改了将数据存储为字符串然后反序列化的方法。问题解决了。

我在原始帖子中添加了一个包含GeMainPage代码的更新。