为了账号安全,请及时绑定邮箱和手机立即绑定

Core 2.0 IServiceCollection 缺少 AddHostedService?

Core 2.0 IServiceCollection 缺少 AddHostedService?

C#
炎炎设计 2022-01-09 10:26:25
尝试使用:启动.cspublic void ConfigureServices(IServiceCollection services) {  services.AddHostedService<LifetimeEvents>();    .    .    .}LifeTimeEvents 类从 IHostedService 继承。我收到此错误:'IServiceCollection' does not contain a definition for 'AddHostedService' and no extension method 'AddHostedService' accepting a first argument of type 'IServiceCollection' could be found (are you missing a using directive or an assembly reference?)我似乎找不到要使用的正确命名空间或要包含的 nuget 包以使其正常工作,但它在 .NET Core 2.1 中开箱即用,这在 .NET Core 2.0 中是否不可用?有什么办法让它工作吗?更新:作为一种解决方法,我将代码更改为使用:启动.cspublic void ConfigureServices(IServiceCollection services) {  services.AddSingleton<LifetimeEvents>();    .    .    .}public void Configure(IApplicationBuilder appBuilder, IHostingEnvironment envHost, LifetimeEvents appEvents)  {  appEvents.StartAsync(new CancellationToken(false));    .    .    .}这似乎已经完成了这项工作。没有回答我最初的问题,我不确定它是多么“最佳实践”,但它确实让我开始重构这个 .NET Core 2.0 应用程序。
查看完整描述

2 回答

?
慕妹3242003

TA贡献1824条经验 获得超6个赞

这只是在 .NET Core 2.0 中不可用吗?


ServiceCollectionHostedServiceExtensions.AddHostedService(IServiceCollection) 方法如 API 参考中所示


适用于

ASP.NET Core

2.1


但是源代码可以在 GitHub 上找到。您可以轻松地在那里查看并将本地版本复制到您的 2.0 项目


namespace Microsoft.Extensions.DependencyInjection

{

    public static class ServiceCollectionHostedServiceExtensions

    {

        /// <summary>

        /// Add an <see cref="IHostedService"/> registration for the given type.

        /// </summary>

        /// <typeparam name="THostedService">An <see cref="IHostedService"/> to register.</typeparam>

        /// <param name="services">The <see cref="IServiceCollection"/> to register with.</param>

        /// <returns>The original <see cref="IServiceCollection"/>.</returns>

        public static IServiceCollection AddHostedService<THostedService>(this IServiceCollection services)

            where THostedService : class, IHostedService

            => services.AddTransient<IHostedService, THostedService>();

    }

}


理想情况下,您可以将项目更新到 2.1,以便扩展可用。


查看完整回答
反对 回复 2022-01-09
?
收到一只叮咚

TA贡献1821条经验 获得超5个赞

我相信你正在寻找这个


https://blogs.msdn.microsoft.com/cesardelatorre/2017/11/18/implementing-background-tasks-in-microservices-with-ihostedservice-and-the-backgroundservice-class-net-core-2-x/


我对自己做了一个 2 小时自称获奖的黑客马拉松,以了解这一点。


https://github.com/nixxholas/nautilus


您可以在此处参考注入并从那里实现摘要。


许多 MVC 项目并不真正需要操作持久的后台任务。这就是为什么您看不到它们通过模板融入新项目的原因。最好为开发人员提供一个可以点击并继续使用的界面。


此外,关于为此类后台任务打开该套接字连接,我还没有为此建立解决方案。据我所知/所做的,我只能将有效负载广播到连接到我自己的 socketmanager 的客户端,所以你必须在别处寻找。如果 IHostedService 中有任何关于 websocket 的信息,我肯定会发出哔哔声。


好吧,无论如何,这就是发生的事情。


把它放在你的项目中的某个地方,它更像是一个让你重载以创建自己的任务的界面


/// Copyright(c) .NET Foundation.Licensed under the Apache License, Version 2.0.

    /// <summary>

    /// Base class for implementing a long running <see cref="IHostedService"/>.

    /// </summary>

    public abstract class BackgroundService : IHostedService, IDisposable

    {

        protected readonly IServiceScopeFactory _scopeFactory;

        private Task _executingTask;

        private readonly CancellationTokenSource _stoppingCts =

                                                       new CancellationTokenSource();


        public BackgroundService(IServiceScopeFactory scopeFactory) {

            _scopeFactory = scopeFactory;

        }


        protected abstract Task ExecuteAsync(CancellationToken stoppingToken);


        public virtual Task StartAsync(CancellationToken cancellationToken)

        {

            // Store the task we're executing

            _executingTask = ExecuteAsync(_stoppingCts.Token);


            // If the task is completed then return it,

            // this will bubble cancellation and failure to the caller

            if (_executingTask.IsCompleted)

            {

                return _executingTask;

            }


            // Otherwise it's running

            return Task.CompletedTask;

        }


        public virtual async Task StopAsync(CancellationToken cancellationToken)

        {

            // Stop called without start

            if (_executingTask == null)

            {

                return;

            }


            try

            {

                // Signal cancellation to the executing method

                _stoppingCts.Cancel();

            }

            finally

            {

                // Wait until the task completes or the stop token triggers

                await Task.WhenAny(_executingTask, Task.Delay(Timeout.Infinite,

                                                              cancellationToken));

            }

        }


        public virtual void Dispose()

        {

            _stoppingCts.Cancel();

        }

    }

这是您实际使用它的方法


public class IncomingEthTxService : BackgroundService

    {

        public IncomingEthTxService(IServiceScopeFactory scopeFactory) : base(scopeFactory)

        {

        }


        protected override async Task ExecuteAsync(CancellationToken stoppingToken)

        {


            while (!stoppingToken.IsCancellationRequested)

            {

                using (var scope = _scopeFactory.CreateScope())

                {

                    var dbContext = scope.ServiceProvider.GetRequiredService<NautilusDbContext>();


                    Console.WriteLine("[IncomingEthTxService] Service is Running");


                    // Run something


                    await Task.Delay(5, stoppingToken);

                }

            }

        }

    }

如果你注意到了,那里有一个奖金。您必须使用服务范围才能访问数据库操作,因为它是单例的。


注入你的服务


// Background Service Dependencies

            services.AddSingleton<IHostedService, IncomingEthTxService>();


查看完整回答
反对 回复 2022-01-09
  • 2 回答
  • 0 关注
  • 254 浏览

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号