abpvnext中使用后台工作者BackgroundWorker创建定时任务,实现定时从更新要缓存的数据到redis

news/2024/5/19 23:24:28 标签: .netcore

abpvnext中使用后台工作者BackgroundWorker创建定时任务,实现定时从更新需要缓存的数据到redis

一、创建后台工作者

1、创建后台工作者

在application层 创建后台工作者类,内容如下 ,其中TAHM为我的自定义前缀名。 这个类的DoWorkAsync方法就是后台工作者默认定时执行的方法,重写这个方法可以填充自己的业务。

注意:为什么我在这个类里加了这么多service注入,那是因为我在第二步创建加载缓存的业务LoadEquipInfoService时,需要这些service,但是我在那边又没法初始化,只能在这个类里初始化,并借助doworkasync方法中的参数

workerContext.ServiceProvider.GetRequiredService<ILoadEquipInfoService>(); 来获取我的缓存业务接口对象,然后他需要的service初始化值就传递过去。

其实简单点理解,就是这个后台工作者类相当于一个控制器,DoWorkAsync相当于一个控制器暴露接口,后台工作者使用的service中需要注入和初始化的对象都要从这里通过workerContext来实现。

using JQ.TAHM.Application.Contracts;
using JQ.TAHM.Domain.Shared;
using JQ.TAHM.Domain;
using Microsoft.Extensions.DependencyInjection; 
using Volo.Abp.BackgroundWorkers;
using Volo.Abp.Caching;
using Volo.Abp.TenantManagement;
using Volo.Abp.Threading;

namespace JQ.TAHM.Application;

public class TAHMBackgroundWorker : AsyncPeriodicBackgroundWorkerBase
    {

        private readonly IDistributedCache<List<PageTahmCJ_descOutput>, string> _cj_descCache;
        private readonly IDistributedCache<List<TahmCJ_wd_tmpDto>, string> _cj_wd_tmpCache;
        private readonly ITenantRepository _tenantrepository;
        private readonly ITahmCJ_descAppService _cj_descappService;
        private readonly ITahmCJ_wd_tmpAppService _cj_wd_tmpappService;
        public TAHMBackgroundWorker(
             AbpAsyncTimer timer,
             IServiceScopeFactory serviceScopeFactory, IDistributedCache<List<PageTahmCJ_descOutput>, string> cj_descCache, IDistributedCache<List<TahmCJ_wd_tmpDto>, string> cj_wd_tmpCache, ITenantRepository tenantrepository, ITahmCJ_descAppService cj_descappService, ITahmCJ_wd_tmpAppService cj_wd_tmpappService
         ) : base(
             timer,
             serviceScopeFactory)
        { 
            Timer.Period = 20000; //此行为任务执行间隔时间,单位毫秒
            //下面5行变量初始化的代码可以不要,这是我的业务需要
            _cj_descCache = cj_descCache;
            _cj_wd_tmpCache = cj_wd_tmpCache;
            _tenantrepository = tenantrepository;
            _cj_descappService = cj_descappService;
            _cj_wd_tmpappService = cj_wd_tmpappService;

        } 
        protected async override Task DoWorkAsync(PeriodicBackgroundWorkerContext workerContext)
        { 
            //这个方法必须重写实现,因为abp默认就会定时执行这个方法.
            //初始可以写Console.WriteLine($"DoWorkAsync执行一次"); 
            //的日志行测试是后台工作者否设定成功。
            var loadEquipInfoService = workerContext
                .ServiceProvider 
                .GetRequiredService<ILoadEquipInfoService>(); 

            await loadEquipInfoService.LoadEquipInfoWithCache(); 
            
        } 
    }

2、注册后台工作者

在HttpApi.Host层 , HttpApiHostModule类,OnApplicationInitialization方法下写入注册后台工作者的代码。我的代码行如下:

context.AddBackgroundWorkerAsync<TAHMBackgroundWorker>();

二、定时更新缓存数据到redis

  1. 做好redis配置

我的HttpApi.Host层,appsettings.json配置代码如下:

"Redis": {

"Configuration": "IP:端口,password=密码,defaultDatabase=2(默认数据库号根据自己实际情况填写),ssl=false,sslHost=null,connectTimeout=4000,allowAdmin=true"

},

然后在HttpApi.Host层,ttpApiHostModule类中 创建一个缓存配置方法,由重写的ConfigureServices方法调用执行。

 private void ConfigureCache(ServiceConfigurationContext context)
        {
            //为应用程序设置缓存键前缀
            Configure<AbpDistributedCacheOptions>(
                options => { options.KeyPrefix = "TAHM:"; }); 
            //下面两行获取redis连接串配置
            var configuration = context.Services.GetConfiguration();
            var redis = ConnectionMultiplexer.Connect(configuration["Redis:Configuration"]);
            //下面这一段的目的我没猜透,加载redis连接串以提供数据防护?
            context.Services
                .AddDataProtection()
                .PersistKeysToStackExchangeRedis(redis, "TAHM-Protection-Keys");
        }

  1. 在application层创建一个定时执行的业务接口(类)ILoadEquipInfoService,我的类在这个层下面, 接口在 Application.Contracts下面。

这个类就是定时任务的业务执行类了,注意除了继承自己类名本身的接口外,还需要继承ApplicationService, ITransientDependency。 而这个类所注入的所有对象,都需要调用他的主人去初始化传导过来。这也符合abp其他通用appService的使用逻辑。

IDistributedCache对象用来存储缓存键值对。

我是要实现定时从数据表按所属租户取出设备信息缓存到redis,以供其他业务使用,所以首先要取出所有租户。开始用ITenantAppService 来取出租户列表,报错提示没有权限,后来被迫无奈直接用ITenantRepository.GetListAsync取列表。

接口类代码如下:

using JQ.TAHM.Application.Contracts; 
using JQ.TAHM.Domain.Shared;
using Microsoft.Extensions.Caching.Distributed;
using Volo.Abp.Caching; 
using Volo.Abp.TenantManagement;
namespace JQ.TAHM.Application;

public class LoadEquipInfoService : ApplicationService, ITransientDependency, ILoadEquipInfoService
    {
        private readonly IDistributedCache<List<PageTahmCJ_descOutput>, string> _cj_descCache;
        private readonly IDistributedCache<List<TahmCJ_wd_tmpDto>, string> _cj_wd_tmpCache;
        
        private readonly ITahmCJ_descAppService _cj_descappService;
        private readonly ITahmCJ_wd_tmpAppService _cj_wd_tmpappService;
        private readonly ITenantAppService _tenantAppService;
        private readonly ITenantRepository _tenantrepository;
        public LoadEquipInfoService(IDistributedCache<List<PageTahmCJ_descOutput>, string> cj_descCache, IDistributedCache<List<TahmCJ_wd_tmpDto>, string> cj_wd_tmpCache, ITenantRepository tenantrepository, ITahmCJ_descAppService cj_descapp, ITahmCJ_wd_tmpAppService cj_wd_tmpapp )
        {
            _tenantrepository = tenantrepository;
            _cj_descappService =    cj_descapp;
            _cj_descCache = cj_descCache;
            _cj_wd_tmpCache = cj_wd_tmpCache;
            _cj_wd_tmpappService = cj_wd_tmpapp;
        }

        public async Task LoadEquipInfoWithCache()
        {
            NoPageTahmCJ_descInput mydescFilter = new NoPageTahmCJ_descInput(); 
            var allTenant= await _tenantrepository.GetListAsync(null,1000000,0,null);

            foreach (var tenant in allTenant)
            {
                mydescFilter.TenantId = tenant.Id.ToString();
                await GetListNoPageAsync(mydescFilter);
            }
            await Task.CompletedTask;
        }

        public async Task<List<PageTahmCJ_descOutput>> GetListNoPageAsync(NoPageTahmCJ_descInput mydescInput)
        {
            var mycachlist = new List<PageTahmCJ_descOutput>();
            var mydescList = _cj_descappService.GetListNoPageAsync(mydescInput);
            var tenantKey = mydescInput.TenantId + "_Desc";
             var cache_option = new DistributedCacheEntryOptions{ AbsoluteExpiration = DateTimeOffset.Now.AddSeconds(20)};  //缓存有效间隔设置

            //强行设置 ,每次更新缓存里的值
            await _cj_descCache.SetAsync(
                tenantKey, 
                await mydescList,
                cache_option
            );


            //查看缓存对象中是否有对应键值的内容存储在,不存在就加入list数据,这个方法有返回list,但是他不会每次都更新缓存里的值。 其实有了上面的SetAsync,这里就可以不需要了
            mycachlist = await _cj_descCache.GetOrAddAsync(
                tenantKey, //缓存键名
                //下面这一行还不能改为同步,只能用异步函数然后await变量值
                async () => await mydescList,
                () =>cache_option
            );

            return mycachlist;
        }
 
    }

接口代码如下:

namespace JQ.TAHM.Application.Contracts;

public interface ILoadEquipInfoService: IApplicationService
{ 
    Task LoadEquipInfoWithCache();
}

以上是我为了实现定时从数据库取出数据缓存到redis的基本思路代码,LoadEquipInfoService其实还有其他数据需要加载到redis,为避免内容太多看不懂,本文就写了一个数据表的redis缓存业务。


http://www.niftyadmin.cn/n/172936.html

相关文章

【C++进阶】哈希表详解

文章目录哈希表是什么&#xff1f;哈希表的底层哈希表的插入哈希冲突闭散列线性探测二线探测开散列(哈希桶)哈希表的删除闭散列删除开散列删除哈希表的查找扩容问题闭散列扩容问题开散列扩容问题哈希表是什么&#xff1f; 哈希表是一种常见的数据结构&#xff0c;英文名是hash…

redis分布式锁及其应用

java锁的回顾&#xff1a;我们在学习java线程的时候都知道&#xff0c;当有多个线程访问共享资源的时候有可能会产生线程安全的问题&#xff0c;那解决的办法大多数就是使用锁了。java中加锁的方式有很多&#xff0c;例如使用synchronized关键字&#xff0c;或者ReentrantLock可…

[NOIP2003 普及组] 乒乓球

题目[NOIP2003 普及组] 乒乓球题目背景国际乒联现在主席沙拉拉自从上任以来就立志于推行一系列改革&#xff0c;以推动乒乓球运动在全球的普及。其中 11 1111 分制改革引起了很大的争议&#xff0c;有一部分球员因为无法适应新规则只能选择退役。华华就是其中一位&#xff0c;他…

Python pip 命令清除Python包缓存文件

Python pip 命令清除Python包缓存文件命令行窗口中安装包命令调用方式示例使用pip命令清除Python包缓存文件命令行窗口中安装包命令 关于如何打开Windows 操作系统下的命令行窗口可以查看这篇------如何在Windows命令行中打开Python自带shell。 调用方式 pip install packag…

潜在因子模型(Latent Factor Model)差分隐私+高斯噪声和证明

潜在因子模型&#xff08;Latent Factor Model&#xff09;是一种常见的数据降维技术&#xff0c;可以将高维数据表示为低维特征空间中的因子分解形式。假设我们有 nnn 个用户和 mmm 个物品&#xff0c;每个用户对某些物品有评分行为&#xff0c;那么可以用潜在因子模型来表示这…

重构条件-Replace Nested Conditional with Guard Clauses以卫语句取代嵌套条件式五

重构条件-Replace Nested Conditional with Guard Clauses以卫语句取代嵌套条件式五 1.以卫语句取代嵌套条件式 1.1.使用场景 函数中的条件逻辑使人难以看清正常的执行路径。使用卫语句表现所有特殊情况 根据我的经验&#xff0c;条件式通常有两种呈现形式。第一种形式是&a…

Python-模块和包

1.模块 概念 在Python中&#xff0c;模块是一种组织Python代码的方式。一个模块(py文件)可以包含多个函数、类、变量等Python对象&#xff0c;可以被其他Python程序导入和使用。Python标准库中包含了很多有用的模块&#xff0c;如math、random、os等。此外&#xff0c;Python还…

go语言通信理解

相信大家都在官网看到这样一段话&#xff0c;“不要通过共享内存来通信&#xff0c;要通过通信开共享内存”。要理解这段话&#xff0c;首先要复习进程间通信和线程间通信的方式&#xff1a;Linux进程间通信&#xff1a;管道&#xff08;匿名管道和有名管道&#xff09;、信号、…