Entity Framework использует много памяти - PullRequest
12 голосов
/ 15 мая 2011

Вот изображение из профилировщика памяти ANTS.Он видит, что в памяти много объектов.Как я могу найти то, что я делаю неправильно?

ANTS memory profiler

**UPDATE**

Вот мои классы репозитория:

public class Repository<T> : IRepository<T> where T : class, IDataEntity
    {
        ObjectContext _context;
        IObjectSet<T> _objectSet;

        readonly string _entitySetName;
        readonly string[] _keyNames;

        private ObjectContext Context
        {
            get
            {
                if (_context == null)
                {
                    _context = GetCurrentUnitOfWork<EFUnitOfWork>().Context;
                }
                return _context;
            }
        }

        private IObjectSet<T> ObjectSet
        {
            get
            {
                if (_objectSet == null)
                {
                    _objectSet = this.Context.CreateObjectSet<T>();
                }
                return _objectSet;
            }
        }

        public TUnitOfWork GetCurrentUnitOfWork<TUnitOfWork>() where TUnitOfWork : IUnitOfWork
        {
            return (TUnitOfWork)UnitOfWork.Current;
        }

        public virtual IEnumerable<T> GetQuery()
        {
            return ObjectSet;
        }

        public virtual IEnumerable<T> GetQuery(params Expression<Func<T, object>>[] includes)
        {
            return ObjectSet.IncludeMultiple(includes);
        }

        public virtual IEnumerable<T> GetQuery(
            IEnumerable<Expression<Func<T, bool>>> filters,
            Func<IQueryable<T>, IOrderedQueryable<T>> orderBy,
            IEnumerable<Expression<Func<T, object>>> includes)
        {
            IQueryable<T> _query = ObjectSet;

            if (filters != null)
            {
                foreach (var filter in filters)
                {
                    _query = _query.Where(filter);
                }
            }

            if (includes != null && includes.Count() > 0)
            {
                _query = _query.IncludeMultiple(includes.ToArray());
            }

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

            return _query;
        }

        public virtual IPaged<T> GetQuery(
            IEnumerable<Expression<Func<T, bool>>> filters,
            Func<IQueryable<T>, IOrderedQueryable<T>> orderBy,
            int pageNumber, int pageSize,
            IEnumerable<Expression<Func<T, object>>> includes)
        {
            IQueryable<T> _query = ObjectSet;

            if (filters != null)
            {
                foreach (var filter in filters)
                {
                    _query = _query.Where(filter);
                }
            }

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

            IPaged<T> page = new Paged<T>(_query, pageNumber, pageSize, includes);

            return page;
        }

        public virtual void Insert(T entity)
        {
            this.ObjectSet.AddObject(entity);
        }

        public virtual void Delete(T entity)
        {
            if (entity is ISoftDeletable)
            {
                ((ISoftDeletable)entity).IsDeleted = true;
                //Update(entity);
            }
            else
            {
                this.ObjectSet.DeleteObject(entity);
            }
        }

        public virtual void Attach(T entity)
        {
            ObjectStateEntry entry = null;
            if (this.Context.ObjectStateManager.TryGetObjectStateEntry(entity, out entry) == false)
            {
                this.ObjectSet.Attach(entity);
            }
        }

        public virtual void Detach(T entity)
        {
            ObjectStateEntry entry = null;
            if (this.Context.ObjectStateManager.TryGetObjectStateEntry(entity, out entry) == true)
            {
                this.ObjectSet.Detach(entity);
            }
        }
    }

Теперь, если у меня есть класс Aкоторый содержит записи из таблицы A, я также создаю класс:

public class ARepository:BaseRepository<A> {
// Implementation of A's queries and specific db operations
}

Вот мой класс EFUnitOfWork:

public class EFUnitOfWork : IUnitOfWork, IDisposable
{
    public ObjectContext Context { get; private set; }

    public EFUnitOfWork(ObjectContext context)
    {
        Context = context;
        context.ContextOptions.LazyLoadingEnabled = true;
    }

    public void Commit()
    {
        Context.SaveChanges();
    }

    public void Dispose()
    {
        if (Context != null)
        {
            Context.Dispose();
        }
        GC.SuppressFinalize(this);
    }
}

И класс UnitOfWork:

public static class UnitOfWork
{
    private const string HTTPCONTEXTKEY = "MyProj.Domain.Business.Repository.HttpContext.Key";

    private static IUnitOfWorkFactory _unitOfWorkFactory;
    private static readonly Hashtable _threads = new Hashtable();

    public static void Commit()
    {
        IUnitOfWork unitOfWork = GetUnitOfWork();
        if (unitOfWork != null)
        {
            unitOfWork.Commit();
        }
    }

    public static IUnitOfWork Current 
    {
        get
        {
            IUnitOfWork unitOfWork = GetUnitOfWork();
            if (unitOfWork == null)
            {
                _unitOfWorkFactory = ObjectFactory.GetInstance<IUnitOfWorkFactory>();
                unitOfWork = _unitOfWorkFactory.Create();
                SaveUnitOfWork(unitOfWork);
            }
            return unitOfWork;
        }
    }

    private static IUnitOfWork GetUnitOfWork()
    {
        if (HttpContext.Current != null)
        {
            if (HttpContext.Current.Items.Contains(HTTPCONTEXTKEY))
            {
                return (IUnitOfWork)HttpContext.Current.Items[HTTPCONTEXTKEY];
            }
            return null;
        }
        else
        {
            Thread thread = Thread.CurrentThread;
            if (string.IsNullOrEmpty(thread.Name))
            {
                thread.Name = Guid.NewGuid().ToString();
                return null;
            }
            else
            {
                lock (_threads.SyncRoot)
                {
                    return (IUnitOfWork)_threads[Thread.CurrentThread.Name];
                }
            }
        }
    }

    private static void SaveUnitOfWork(IUnitOfWork unitOfWork)
    {
        if (HttpContext.Current != null)
        {
            HttpContext.Current.Items[HTTPCONTEXTKEY] = unitOfWork;
        }
        else
        {
            lock(_threads.SyncRoot)
            {
                _threads[Thread.CurrentThread.Name] = unitOfWork;
            }
        }
    }
}

ЗдесьВот как я использую это:

 public class TaskPriceRepository : BaseRepository<TaskPrice>
    {
        public void Set(TaskPrice entity)
        {
            TaskPrice taskPrice = GetQuery().SingleOrDefault(x => x.TaskId == entity.TaskId);
            if (taskPrice != null)
            {
                CommonUtils.CopyObject<TaskPrice>(entity, ref taskPrice);
            }
            else
            {
                this.Insert(entity);
            }
        }
    }

public class BranchRepository : BaseRepository<Branch>
{
    public IList<Branch> GetBranchesList(Guid companyId, long? branchId, string branchName)
    {
        return Repository.GetQuery().
            Where(b => companyId == b.CompanyId).
            Where(b => b.IsDeleted == false).
            Where(b => !branchId.HasValue || b.BranchId.Equals(branchId.Value)).
            Where(b => branchName == null || b.BranchName.Contains(branchName)).
            ToList();
    }
}

[WebMethod]
public void SetTaskPrice(TaskPriceDTO taskPrice)
{
    TaskPrice tp = taskPrice.ToEntity();
    TaskPriceRepository rep = new TaskPriceRepository();
    rep.Set(tp);
    UnitOfWork.Commit();
}

[WebMethod]
public IList<Branch> GetBranchesList()
{
    BranchRepository rep = new BranchRepository();
    return rep.GetBranchesList(m_User.UserCompany.CompanyId, null, null).ToList();
}

Надеюсь, этого достаточно, чтобы помочь мне решить проблему.Спасибо.

UPDATE 2
Существует также UnitOfWorkFactory, которая инициализирует UnitOfWork:

public class UnitOfWorkFactory : IUnitOfWorkFactory
{
    private static Func<ObjectContext> _objectContextDelegate;
    private static readonly Object _lockObject = new object();

    public static void SetObjectContext(Func<ObjectContext> objectContextDelegate)
    {
        _objectContextDelegate = objectContextDelegate;
    }

    public IUnitOfWork Create()
    {
        ObjectContext context;
        lock (_lockObject)
        {
             context = _objectContextDelegate();
        }
        return new EFUnitOfWork(context);
    }
}

Чтобы использовать это, при запуске приложения я использую Structuremap:

  ObjectFactory.Initialize(x =>
        {
            x.For<IUnitOfWorkFactory>().Use<UnitOfWorkFactory>();
            x.For(typeof(IRepository<>)).Use(typeof(Repository<>));
        });

Ответы [ 4 ]

15 голосов
/ 15 мая 2011

У меня есть догадка, что вы не располагаете контекстом.
Я предлагаю использовать контекст всякий раз, когда вы взаимодействуете с базой данных.

Используйте оператор using при создании контекста.

[Изменить]

Насколько я вижу, вы кешируете и не утилизируете свой EFUnitOfWork объект. Это одноразовый, что правильно, но я не вижу, когда называется одноразовый. Похоже, у вас есть ссылка на контекст для всех приложений.
Более того, вы создаете и удерживаете один контекст на поток, что сделает его еще хуже.

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

Кроме того, ваш дизайн может быть слишком сложным.

На вашем месте я бы:

  • Найдите способ распоряжаться контекстом с использованием текущего кода в качестве краткосрочного решения
  • Упростите дизайн, как долгосрочное решение

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

5 голосов
/ 15 мая 2011

Мне приходит в голову пара вещей:

  • Вы, вероятно, не избавляетесь от ObjectContext.Убедитесь, что все коды вашей базы данных находятся в пределах using(var context = CreateObjectContext()) block
  • . У вас есть N-уровневая архитектура, и вы передаете сущности из уровня доступа к данным на верхний уровень без отсоединения сущностей от ObjectContext.Вам необходимо вызвать ObjectContext.Detach (...)
  • Скорее всего, вы возвращаете полную коллекцию сущностей, а не возвращаете одну единицу для отдельных операций Get.Например, у вас есть запросы типа from customer in context.Customers select customer вместо from customer in context.Customers select customer.FirstOrDefault()

Мне было трудно заставить Entity Framework работать в N-уровневом приложении.Он просто не подходит для использования в приложениях N-уровня как есть.Только EF 4.0 есть.Вы можете прочитать обо всех моих приключениях по созданию EF 3 в приложении N-уровня.

http://www.codeproject.com/KB/linq/ef.aspx

Отвечает ли это на ваш вопрос?

3 голосов
/ 15 мая 2011

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

0 голосов
/ 26 июня 2019

У меня была такая же проблема в классе, который использует внедрение зависимостей, поэтому опция using() не была альтернативой. Мое решение состояло в том, чтобы добавить DbContextOptions<Context> к конструктору и как личное поле к классу. Затем вы можете позвонить

_db.Dispose();
_db = new BlockExplorerContext(_dBContextOptions);

в соответствующее время. Это исправило мою проблему, когда у меня не хватало оперативной памяти, а приложение было убито ОС.

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