C# 提前异步加载数据
前言
在我们应用程序中,如果有较大的数据需要从数据库或者本地读取,且是一次性的话,可以提前获取数据并缓存在内存中。
一般处理方法:利用应用程序启动到用户使用功能这一段时间,提前加载数据。
问题来了,因数据大小的不固定性,提前加载的速度有快有慢,如果保证不会重复读取数据库、只返回一次结果呢?
Task ConfigureAwait
使用ConfigureAwait可以解决以上问题。详情如下:
程序启动后,在相应的事件中,调用初始化方法
1 private Task<List<EnglishWordInfo>> _wordsCacheDelayTask; 2 /// <summary> 3 /// 提前初始化字典 4 /// 注:通过InitWordInfos提前缓存数据,提升用户体验 5 /// </summary> 6 public void InitWordInfos() 7 { 8 if (_wordsCacheDelayTask == null) 9 { 10 _wordsCacheDelayTask = GetWordsTask(); 11 } 12 }
使用ConfigureAwait异步获取数据
通过ConfigureAwait的使用,不等待结果的返回。
1 /// <summary> 2 /// 延时获取数据 3 /// 注:延时获取,避免界面卡顿 4 /// </summary> 5 /// <returns></returns> 6 private async Task<List<EnglishWordInfo>> GetWordsTask() 7 { 8 return await Task.Run(() => 9 { 10 return EnglishDictService.Instance.GetWords(); 11 }).ConfigureAwait(false); 12 }
添加封装数据延迟任务
如果不是程序启动后立即加载缓存,而是程序使用期间对数据缓存的处理。可以添加如下对缓存延迟任务的封装,保证只获取一次缓存数据。
1 /// <summary> 2 /// 获取数据延迟任务 3 /// 注:如果没有提前延迟任务,则重新获取数据 4 /// </summary> 5 public Task<List<EnglishWordInfo>> WordsCacheDelayTask 6 { 7 set => _wordsCacheDelayTask = value; 8 get 9 { 10 if (_wordsCacheDelayTask == null) 11 { 12 _wordsCacheDelayTask = GetWordsTask(); 13 } 14 15 return _wordsCacheDelayTask; 16 } 17 }
获取缓存数据
通过Task.Result,获取缓存任务的结果,此缓存任务的状态可以是:
- 执行中 -- 则会在原有进度下,继续执行并返回结果
- 已完成 -- 则会返回原有的结果
无论获取多少次,Result只会返回同样的结果,可以把WordsCacheDelayTask.Result看成是一个静态的缓存。
1 /// <summary> 2 /// 获取缓存数据 3 /// </summary> 4 private List<EnglishWordInfo> WordInfosCache => WordsCacheDelayTask.Result;
参考列表:
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 一个奇形怪状的面试题:Bean中的CHM要不要加volatile?
· [.NET]调用本地 Deepseek 模型
· 一个费力不讨好的项目,让我损失了近一半的绩效!
· .NET Core 托管堆内存泄露/CPU异常的常见思路
· PostgreSQL 和 SQL Server 在统计信息维护中的关键差异
· DeepSeek “源神”启动!「GitHub 热点速览」
· 我与微信审核的“相爱相杀”看个人小程序副业
· 上周热点回顾(2.17-2.23)
· 如何使用 Uni-app 实现视频聊天(源码,支持安卓、iOS)
· C# 集成 DeepSeek 模型实现 AI 私有化(本地部署与 API 调用教程)