使用MiniProfiler和EF 5以及Autofac分析DbContext的正确方法

Yaa*_*lis 6 autofac dbcontext mvc-mini-profiler entity-framework-5

MiniProfiler网站给出了生成实体框架下面的代码ObjectContext:

public static MyModel Get()
{
  var conn =  new StackExchange.Profiling.Data.EFProfiledDbConnection(GetConnection(), MiniProfiler.Current);
  return ObjectContextUtils.CreateObjectContext<MyModel>(conn); // resides in the MiniProfiler.EF nuget pack
}
Run Code Online (Sandbox Code Playgroud)

但是,使用Entity Framework 5,我没有使用ObjectContext- 而是我使用的是DbContext.我不能在这里插入模型名称,因为该CreateObjectContext<T>()方法需要T是类型ObjectContext.(出于同样的原因,此答案中给出的代码也不起作用).

另外,我使用autofac来初始化我的Db连接.这是使用以下内容注册的(MyData=我的EF DataContext的名称):

Builder.RegisterType<MyData>().As<DbContext>().InstancePerHttpRequest();
Run Code Online (Sandbox Code Playgroud)

因此结合两部分:如何使用autofac初始化与MiniProfiler.EF绑定的DbContext?如果这是不可能的,至少我怎么做第一部分(为MiniProfiler.EF创建一个工厂方法来返回DbContext)?

khe*_*ang 11

我刚刚开始工作:

public static class DbContextUtils
{
    private const BindingFlags PrivateInstance = BindingFlags.NonPublic | BindingFlags.Instance;

    public static T CreateDbContext<T>() where T : DbContext
    {
        return CreateDbContext<T>(GetProfiledConnection<T>());
    }

    public static T CreateDbContext<T>(this DbConnection connection) where T : DbContext
    {
        var workspace = new MetadataWorkspace(new[] { "res://*/" }, new[] { typeof(T).Assembly });
        var factory = DbProviderServices.GetProviderFactory(connection);

        var itemCollection = workspace.GetItemCollection(DataSpace.SSpace);
        var providerFactoryField = itemCollection.GetType().GetField("_providerFactory", PrivateInstance);
        if (providerFactoryField != null) providerFactoryField.SetValue(itemCollection, factory);

        var ec = new EntityConnection(workspace, connection);

        return CtorCache<T, DbConnection>.Ctor(ec);
    }

    public static DbConnection GetProfiledConnection<T>() where T : DbContext
    {
        var dbConnection = ObjectContextUtils.GetStoreConnection("name=" + typeof(T).Name);
        return new EFProfiledDbConnection(dbConnection, MiniProfiler.Current);
    }

    internal static class CtorCache<TType, TArg> where TType : class
    {
        public static readonly Func<TArg, TType> Ctor;
        static CtorCache()
        {
            var argTypes = new[] { typeof(TArg) };
            var ctor = typeof(TType).GetConstructor(argTypes);
            if (ctor == null)
            {
                Ctor = x => { throw new InvalidOperationException("No suitable constructor defined"); };
            }
            else
            {
                var dm = new DynamicMethod("ctor", typeof(TType), argTypes);
                var il = dm.GetILGenerator();
                il.Emit(OpCodes.Ldarg_0);
                il.Emit(OpCodes.Newobj, ctor);
                il.Emit(OpCodes.Ret);
                Ctor = (Func<TArg, TType>)dm.CreateDelegate(typeof(Func<TArg, TType>));
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

它基于MiniProfiler中的ObjectContextUtils代码.

你这样使用它:

builder.Register(c => DbContextUtils.CreateDbContext<MyData>()).As<DbContext>().InstancePerHttpRequest();
Run Code Online (Sandbox Code Playgroud)

这个解决方案需要DbContext有一个构造函数,它接受DbConnection并传递给base,如下所示:

public MyData(DbConnection connection)
    : base(connection, true)
{
}
Run Code Online (Sandbox Code Playgroud)

  • 谢谢克里斯蒂安......这就像一个冠军! (2认同)

nem*_*esv 6

有一个构造函数中的DbContext类,它采用现有的DbConnection

所以你需要一个新的构造函数来MyData调用基础

public class MyData : DbContext
{
    public MyData(DbConnection existingConnection, bool contextOwnsConnection)
        : base(existingConnection, contextOwnsConnection)
    {
    }

    //..
}
Run Code Online (Sandbox Code Playgroud)

然后您注册MyData使用Register:

builder.Register(c => 
{
   var conn =  new EFProfiledDbConnection(GetConnection(), MiniProfiler.Current);
   return new MyData(conn, true);
}).As<DbContext>().InstancePerHttpRequest();
Run Code Online (Sandbox Code Playgroud)