gpt4 book ai didi

c# - 从 IoC 容器加载重物会阻塞 UI

转载 作者:太空宇宙 更新时间:2023-11-03 23:43:26 24 4
gpt4 key购买 nike

我们有一个 WPF 应用程序(.NET 4.0,由于与 Windows Server 2003 和 XP 的兼容性而无法更改)使用 BCL 来支持异步/等待。对于 DI 和方面,我们使用 CaSTLe IoC,为了访问 Oracle 数据库,我们使用 NHibernate。我们的问题如下:

我已经实现了一个 UoW 模式,在某些时候,它必须接收一个 ISessionFactory。由于我们使用 async/await ,用户调用的第一个业务操作将首次加载 ISessionFactory 单例,由于映射(即使我将该信息存储在文件中),这可能需要几秒钟,并且因为异步/等待它将在 UI 上执行,阻止应用程序。我想要实现的是在单独的线程上加载 ISessionFactory,试图防止 UI 阻塞。因为第一次加载可能发生在任何屏幕、任何给定时间,我相信 IoC 容器应该是放置此逻辑的最佳位置,但我未能阻止 UI 阻塞。

这里是正常的注册:

    Component.For<ISessionFactory>().UsingFactoryMethod(
k =>
Fluently.Configure()
.Database(
() => OracleClientConfiguration.Oracle10.ConnectionString(
c => c.Is(k.Resolve<ISifarmaConnectionString>().Value))
#if DEBUG
.ShowSql().FormatSql()
#endif
)
.Mappings(m => m.FluentMappings.AddFromAssemblyOf<ISifarmaUnitOfWork>())
#if DEBUG
.ExposeConfiguration(c => c.SetInterceptor(new SqlStatementInterceptor()))
#endif
.BuildSessionFactory()).LifestyleSingleton(),

这是尝试过的方法(因为我返回的是具体类型而不是 Task<ISessionFactory> 我不能在这里等待,这显然用 Task.WaitAll(task) 阻塞了 UI):

    Component.For<ISessionFactory>().UsingFactoryMethod(
k =>
{
var task =
Task.Factory.StartNew(
() =>
Fluently.Configure()
.Database(
() => OracleClientConfiguration.Oracle10.ConnectionString(
c => c.Is(k.Resolve<ISifarmaConnectionString>().Value))
#if DEBUG
.ShowSql().FormatSql()
#endif
)
.Mappings(m => m.FluentMappings.AddFromAssemblyOf<ISifarmaUnitOfWork>())
#if DEBUG
.ExposeConfiguration(c => c.SetInterceptor(new SqlStatementInterceptor()))
#endif
.BuildSessionFactory());
Task.WaitAll(task);
return task.Result;
}).LifestyleSingleton(),

有没有什么方法可以在不阻塞 UI 的情况下做到这一点?我还尝试在应用程序启动时创建一个任务,使 container.Resolve<ISessionFactory>()但如果用户在登录屏幕上的速度足够快,则对象尚未加载。

最佳答案

您应该做的是为 ISessionFactory 创建一个依赖于 Lazy<ISessionFactory> 的代理类。这个代理可以注入(inject)任何需要 ISessionFactory 的人,你可以在初始化期间在后台线程中自己触发 ISessionFactory 的创建:

public class LazySessionFactoryProxy : ISessionFactory
{
private readonly Lazy<ISessionFactory> factory;

public LazySessionFactoryProxy(Lazy<ISessionFactory> factory) {
this.factory = factory;
}

public ISession OpenSession() {
return this.factory.Value.OpenSession();
}
}

这可以按如下方式注册:

var lazy = new Lazy<ISessionFactory>(() => ...);

container.Register(Component.For<ISessionFactory>()
.Instance(new LazySessionFactoryProxy(lazy)));

Task.Factory.StartNew(() => lazy.Value);

通过在组合根中创建代理类,您可以让应用程序的其余部分忽略 session 工厂中的性能瓶颈。向每个消费者注入(inject) Lazy<ISessionFactory>Task<ISessionFactory> 意味着您需要在整个应用程序中进行彻底的更改(违反 OCP )并且意味着您正在从抽象中泄漏实现细节(创建特定 session 工厂的成本很高)(违规) DIP )。

您甚至可以创建一个 LazySessionFactoryProxy,它会在某些代码调用 OpenSession 时显示等待屏幕,并会在工厂初始化时自动关闭它。有趣的是,这一切都是可能的,而无需更改应用程序中的一行代码;只需更改组合根的接线即可。

请注意,DI 容器不支持“异步解析”的问题,但异步解析根本没有用。构建你的 object graphs should be really fast 因为 injection constructors should be simple 和缓慢的部分应该按照我展示的方式隔离,而不是用它污染应用程序。

关于c# - 从 IoC 容器加载重物会阻塞 UI,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28298357/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com