c# - MVC ASP.NET 正在使用大量内存

标签 c# asp.net asp.net-mvc entity-framework memory

如果我只是浏览应用程序上的一些页面,它大约有 500MB。这些页面中的许多页面都访问数据库,但此时,我只有大约几行,每个 10 个表,主要存储字符串和一些小于 50KB 的小图标。

真正的问题发生在我下载文件时。该文件大约为 140MB,并以 varbinary(MAX) 的形式存储在数据库中。内存使用量在一瞬间突然上升到 1.3GB,然后又回落到 1GB。该操作的代码在这里:

public ActionResult DownloadIpa(int buildId)
{
    var build = _unitOfWork.Repository<Build>().GetById(buildId);
    var buildFiles = _unitOfWork.Repository<BuildFiles>().GetById(buildId);
    if (buildFiles == null)
    {
        throw new HttpException(404, "Item not found");
    }

    var app = _unitOfWork.Repository<App>().GetById(build.AppId);
    var fileName = app.Name + ".ipa";

    app.Downloads++;
    _unitOfWork.Repository<App>().Update(app);
    _unitOfWork.Save();

    return DownloadFile(buildFiles.Ipa, fileName);
}

private ActionResult DownloadFile(byte[] file, string fileName, string type = "application/octet-stream")
{
    if (file == null)
    {
        throw new HttpException(500, "Empty file");
    }

    if (fileName.Equals(""))
    {
        throw new HttpException(500, "No name");
    }

    return File(file, type, fileName);            
}

在我的本地计算机上,如果我什么都不做,内存使用量将保持在 1GB。如果我然后返回并导航到某些页面,它会回落到 500MB。

在部署服务器上,无论我做什么,第一次下载后它都保持在 1.6GB。我可以通过不断下载文件来强制增加内存使用量,直到达到 3GB,然后又回落到 1.6GB。

在每个 Controller 中,我都重写了 Dispose() 方法:

protected override void Dispose(bool disposing)
{
    _unitOfWork.Dispose();
    base.Dispose(disposing);
}

这里指的是:

public void Dispose()
{
    Dispose(true);
    GC.SuppressFinalize(this);
}

public void Dispose(bool disposing)
{
    if (!_disposed)
    {
        if (disposing)
        {
            _context.Dispose();
        }
    }

    _disposed = true;
}

所以我的工作单元应该在每次处置 Controller 时处置。我正在使用 Unity,并且向 Heirarchical Lifetime Manager 注册了工作单元。

以下是 Profiler 的一些屏幕截图:

enter image description here

enter image description here

enter image description here

我认为这可能是问题所在,或者我走错了路。为什么 Find() 会使用 300MB?

编辑:

存储库:

public class Repository<TEntity> : IRepository<TEntity> where TEntity : class
{
    internal IDbContext Context;
    internal IDbSet<TEntity> DbSet;

    public Repository(IDbContext context)
    {
        Context = context;
        DbSet = Context.Set<TEntity>();
    }

    public virtual IEnumerable<TEntity> GetAll()
    {            
        return DbSet.ToList();
    }

    public virtual TEntity GetById(object id)
    {
        return DbSet.Find(id);
    }

    public TEntity GetSingle(Expression<Func<TEntity, bool>> predicate)
    {
        return DbSet.Where(predicate).SingleOrDefault();
    }

    public virtual RepositoryQuery<TEntity> Query()
    {
        return new RepositoryQuery<TEntity>(this);
    }

    internal IEnumerable<TEntity> Get(
        Expression<Func<TEntity, bool>> filter = null,
        Func<IQueryable<TEntity>, IOrderedQueryable<TEntity>> orderBy = null,
        List<Expression<Func<TEntity, object>>> includeProperties = null)
    {
        IQueryable<TEntity> query = DbSet;

        if (includeProperties != null)
        {
            includeProperties.ForEach(i => query.Include(i));
        }

        if (filter != null)
        {
            query = query.Where(filter);
        }

        if (orderBy != null)
        {
            query = orderBy(query);
        }

        return query.ToList();
    }

    public virtual void Insert(TEntity entity)
    {
        DbSet.Add(entity);
    }

    public virtual void Update(TEntity entity)
    {
        DbSet.Attach(entity);
        Context.Entry(entity).State = EntityState.Modified;
    }

    public virtual void Delete(object id)
    {
        var entity = DbSet.Find(id);

        Delete(entity);
    }

    public virtual void Delete(TEntity entity)
    {
        if (Context.Entry(entity).State == EntityState.Detached)
        {
            DbSet.Attach(entity);
        }

        DbSet.Remove(entity);
    }
}

编辑 2:

我为各种场景运行 dotMemory,这就是我得到的。

enter image description here

红色圆圈表示有时一页访问会发生多次上升和下降。蓝色圆圈表示下载 40MB 文件。绿色圆圈表示下载 140MB 文件。此外,很多时候,即使页面立即加载,内存使用量也会持续增加几秒钟。

最佳答案

因为文件很大,所以它被分配在大对象堆上,该堆是使用 gen2 集合收集的(您在配置文件中看到,紫色 block 是大对象堆,您会在 10 秒后看到它被收集)。

在您的生产服务器上,您很可能拥有比本地机器更多的内存。因为内存压力较小,收集不会那么频繁,这就解释了为什么它会增加一个更高的数字 - LOH 在收集之前有几个文件。

如果在 MVC 和 EF 中的不同缓冲区中,一些数据也被复制到不安全的 block 中,我一点也不会感到惊讶,这解释了非托管内存增长(EF 的细尖峰,MVC 的宽平台)

最后,一个 500MB 的基线对于一个大型项目来说并不完全令人惊讶(疯狂!但确实如此!)

因此,对于您的问题,为什么它很可能会使用这么多内存的答案是“因为它可以”,或者换句话说,因为执行 gen2 收集没有内存压力,并且下载的文件未使用您的大型对象堆,直到收集驱逐它们,因为您的生产服务器上的内存充足。

这甚至可能不是一个真正的问题:如果内存压力更大,就会有更多的收集,而您会发现内存使用率更低。

至于该怎么做,恐怕你对 Entity Framework 不走运。据我所知,它没有流式 API。顺便说一句,WebAPI 确实允许流式传输响应,但是如果您将整个大对象都放在内存中,这对您没有多大帮助(尽管它可能对一些(由我)未开发的 MVC 部分中的非托管内存有所帮助.

关于c# - MVC ASP.NET 正在使用大量内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25993290/

相关文章:

asp.net - Asp.net 网站位于服务器上时出错

asp.net-mvc - 未找到 Web Api 路由

asp.net-mvc - 模型中的 ASP.NET MVC 绑定(bind)数组

javascript - 在客户端 ASP.NET MVC 上将 List<string> 转换为 Json 数组

c# - 处理队列外的文件,归档并存储在数据库中

c# - 在 VS2012 中获取 TypeInitializationException,但不是 2010

asp.net - 修改通过 javascript 呈现的 Gridview

javascript - 如何设置gridview文本框高度以避免在itemtemplate中滚动

c# - WPF SelectedItem 颜色在列表失去焦点时消失

c# - 该通知没有目标应用程序