.NET 中的 Worker Service 入门介绍
翻译自 Steve Gordon 2020年3月30日的文章 《WHAT ARE .NET WORKER SERVICES?》 [1]
随着 .NET Core 3.0 的发布,ASP.NET 团队引入了一个新的 Worker Service 项目模板,该模板作为 .NET SDK 的一部分发布。在本文中,我将向您介绍这个新模板,以及使用它开发的一些实际的服务示例。
译者注:
请先完成以下准备工作,以便于您理解本文。
1、下载并安装最新的 .NET SDK:https://dotnet.microsoft.com/download
2、命令行运行dotnet new Worker -n "MyService"命令,创建一个 Worker Service 项目。
什么是 .NET Core Worker Service?
Worker Service 是使用模板构建的 .NET 项目,该模板提供了一些有用的功能,可以将常规控制台应用程序变得更加强大。Worker Service 运行于宿主(Host)的概念之上,宿主维护应用程序的生命周期。宿主还提供了一些常见的特性,如依赖注入、日志记录和配置。
Worker Service 通常是长时间运行的服务,执行一些规律发生的工作负载。
Worker Service 的一些例子
- 处理来自队列、服务总线或事件流的消息、事件
- 响应对象、文件存储中的文件更改
- 聚合数据存储中的数据
- 丰富数据提取管道中的数据
- AI/ML 数据集的格式化和清理
还可以开发一个这样的 Worker Service,该服务从头到尾执行一个过程,然后关闭。结合调度程序,便可以支持定期的批处理工作负载。例如,调度程序每隔一小时启动一次服务,完成一些汇总数据的计算,然后关闭。
Worker Service 没有用户界面,也不支持直接的用户交互,它们特别适用于设计微服务架构。在微服务体系结构中,职责通常被划分为不同的、可单独部署的、可伸缩的服务。随着微服务架构的成长和发展,拥有大量的 Worker Service 会变得越来越常见。
Worker Service 模板提供了什么?
完全可以在不使用 Worker Service 模板的情况下开发长时间运行的 Worker Service。在 .NET Core 的早期版本中我是这样做的,使用依赖注入容器手动建立宿主,然后启动我的处理工作负载。
在默认情况下,Worker Service 模板包含了有用的基础组件,比如依赖注入,这样我们就可以集中精力在其上构建业务逻辑。它包含了一个管理应用程序生命周期的宿主。
Worker Service 模板本身是相当基础的,它只包含了三个开箱即用的核心文件。
1. Program.cs
第一个是 Program 类。该类包含 .NET 控制台应用程序所必需的 Main 方法入口点,.NET 运行时期望在启动 .NET 应用程序时在 Program 类中查找此方法。
public class Program
{
public static void Main(string[] args)
{
CreateHostBuilder(args).Build().Run();
}
public static IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
.ConfigureServices((hostContext, services) =>
{
services.AddHostedService<Worker>();
});
}
在 Program 类中,作为 Worker Service 模板一部分的是 CreateHostBuilder 方法,该方法创建一个 IHostBuilder。IHostBuilder 接口定义了一个类型,该类型使用生成器模式生成 IHost 的实例。此模板通过调用 Host 类中的静态 CreateDefaultBuilder 方法来创建一个新的 HostBuilder。
然后,它使用生成器来配置 IHost,该 IHost 被用于运行 Worker Service 应用程序。宿主提供了依赖注入容器和日志记录等功能,就像我们可以在 ASP.NET Core 应用程序中使用的那样。事实上,从 .NET Core 3.0 开始,ASP.NET Core Web 应用程序和 .NET Core Worker Service 都运行在同一 IHost 上的。
默认情况下,它包含了一个服务注册,稍后我会在本文中介绍,暂时不用担心。
从 Main 方法中调用 CreateDefaultBuilder 方法,将构建并立即运行宿主。当 .NET 运行时调用 Main 方法时,应用程序启动,宿主将保持运行,监听标准的关闭信号(例如按下 CTRL+C 键)。
2. appsettings.json
如果您以前使用过 ASP.NET Core,将会非常熟悉 appsettings.json 文件,它是应用程序配置的常见来源之一。宿主被设计为,当启动应用程序时,使用任意已注册的配置提供程序从多个来源加载应用程序配置。其中一种提供程序是从 appsettings.json 加载配置,该文件内容由 JSON 组成,其结构包含表示应用程序配置的键和值。这些值可以随意地定义在对相关配置按逻辑分组成的片段(Sections)内。
在 Worker Service 中,启动时会检查相同的配置源(包括此 appsettings.json 文件和环境变量),并从不同的源构建最终的配置。默认情况下会加载多种默认的提供程序,因此也会加载多种源。如果需要,您也可以自定义宿主用来加载配置数据的提供程序。
{
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft": "Warning",
"Microsoft.Hosting.Lifetime": "Information"
}
}
}
模板中的默认 appsettings 文件中包含日志记录库的配置设置项,默认对 Worker Service 可用。这里的配置是为某些日志记录上下文设置记录级别的。
3. Worker.cs
Worker 是一个您在默认的 ASP.NET Core 项目模板中见不到的新类。它是托管服务与宿主相结合的魔力所在,提供了 Worker Service 的基础。
让我们来看一下它的代码:
public class Worker : BackgroundService
{
private readonly ILogger<Worker> _logger;
public Worker(ILogger<Worker> logger)
{
_logger = logger;
}
protected override async Task ExecuteAsync(CancellationToken stoppingToken)
{
while (!stoppingToken.IsCancellationRequested)
{
_logger.LogInformation("Worker running at: {time}", DateTimeOffset.Now);
await Task.Delay(1000, stoppingToken);
}
}
}
此类从 BackgroundService 抽象基类派生。BackgroundService 类实现了一个名为 IHostedService 的接口。
BackgroundService 包含一个名为 ExecuteAsync 的抽象方法,我们必须在子类中重写该方法,就像 Worker Service 模板中提供的 Worker 类中所做的那样。ExecuteAsync 方法返回一个 Task,在 BackgroundService 内部,期望此 Task 是一些长时间运行的工作负载。该 Task 会被启动并在后台运行。
在内部,宿主将启动 IHostedService 的所有注册实现(包括从 BackgroundService 抽象类派生的类型)。请记住,BackgroundService 为我们实现了 IHostedService。
4. 如何注册托管服务(IHostedService)?
下一个显而易见的问题是,如何注册 IHostedService ? 如果我们返回到 Program.cs 的代码,我们将会找到答案:
public static IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
.ConfigureServices((hostContext, services) =>
{
services.AddHostedService<Worker>();
});
在 ConfigureServices 方法中,可以向依赖注入容器注册类型。AddHostedService 是为 IServiceCollection 定义的一个扩展方法,它允许我们注册一个实现了 IHostedService 的类。
该模板中已将 Worker 类注册为托管服务。
在启动时,宿主将找到已注册的 IHostedService 的所有实例,并按顺序启动它们,此时,它们的长时间运行的工作负载会作为后台任务来运行。
为什么要构建 .NET Core Worker Service?
简单的答案是——何时以及是否需要它们!如果您需要开发一个微服务,它没有用户界面,并执行长时间运行的工作,那么 Worker Service 很可能是一个好的选择。
请记住,Worker Service 的底层只是一个控制台应用程序。该控制台应用程序使用宿主将应用程序转换为运行的服务,直到收到停止的信号。宿主带来了您可能已经熟悉的一些特性,比如依赖关系注入。使用和 ASP.NET Core 中可用的相同的日志记录和配置扩展,使得开发可记录日志信息且需要一些配置的 Worker Service 变得相当轻松。当构建运行在云上的 Worker Service 时,几乎总会存在这种需求。例如,您可能需要为与您的 Worker Service 相交互的任何外部服务提供配置(比如一个队列 URL)。
Worker Service 可用于从现有的 ASP.NET Core 应用程序提取职责,设计新的基于 .NET Core 的微服务。
总结
在本文中,我介绍了 Worker Service 项目模板,以及它的一些潜在用例。我们探索了使用 Worker Service 模板创建的新项目中所包含的三个默认文件。
Worker Service 模板包含哪些文件?
- Program.cs: 控制台应用程序的入口点,创建并运行宿主以管理应用程序生命周期并生成一个长期运行的服务。
- appsettings.json:一个提供应用程序配置值的 JSON 文件。
- Worker.cs:派生自 BackgroundService 基类,用于定义作为后台任务执行的长时间运行的工作负载。
Worker Service 是什么?
- 不需要用户交互的应用程序。
- 使用宿主来维护控制台应用程序的生命周期,直到宿主收到关闭的信号。将控制台应用程序转换为长时间运行的服务。
- 包含和 ASP.NET Core 相同的功能,如依赖注入、日志记录和配置。
- 执行定期和长时间运行的工作负载。
https://www.stevejgordon.co.uk/what-are-dotnet-worker-services WHAT ARE .NET WORKER SERVICES? ︎
.NET 中的 Worker Service 入门介绍的更多相关文章
- .NET Core中的Worker Service
当你想到ASP.NET Core时,可能会想到Web应用程序后端代码,包括MVC和WebAPI.MVC视图和Razor页面还允许使用后端代码生成带有HTML元素的前端UI.全新的Blazor更进一步, ...
- Android入门:Service入门介绍
一.Service介绍 Service类似于Windows中的服务,没有界面,只是在后台运行:而服务不能自己运行,而是需要调用Context.startService(Intent intent);或 ...
- 【iOS开发】在 App 中加入 AdMob 广告 - 入门介绍与编程技巧
前言 虽然在App中加入广告来盈利是比较低级的商业化方式,但对于个人开发者或者小团队开发者来说,做出一个简单易用的免费小工具(举个栗子),在里面加入一些广告,如果用的人多,也是可以带来一些可观的收入的 ...
- .NET Worker Service 如何优雅退出
上一篇文章中我们了解了 .NET Worker Service 的入门知识[1],今天我们接着介绍一下如何优雅地关闭和退出 Worker Service. Worker 类 从上一篇文章中,我们已经知 ...
- .NET Worker Service 添加 Serilog 日志记录
前面我们了解了 .NET Worker Service 的入门知识[1] 和 如何优雅退出 Worker Service [2],今天我们接着介绍一下如何为 Worker Service 添加 Ser ...
- .NET Worker Service 作为 Windows 服务运行及优雅退出改进
上一篇文章我们了解了如何为 Worker Service 添加 Serilog 日志记录,今天我接着介绍一下如何将 Worker Service 作为 Windows 服务运行. 我曾经在前面一篇文章 ...
- .NET Worker Service 部署到 Linux 作为 Systemd Service 运行
上一篇文章我们了解了如何将.NET Worker Service 作为 Windows 服务运行,今天我接着介绍一下如何将 Worker Service 部署到 Linux 上,并作为 Systemd ...
- 学习Mysql过程中拓展的其他技术栈:Docker入门介绍
一.Docker的介绍和安装 1. Docker是什么 百度百科的介绍: Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的Linu ...
- 在 ASP.NET Core和Worker Service中使用Quartz.Net
现在有了一个官方包Quartz.Extensions.Hosting实现使用Quartz.Net运行后台任务,所以把Quartz.Net添加到ASP.NET Core或Worker Service要简 ...
随机推荐
- Cloud Alibabab笔记问世,全网详解仅此一份手慢无
转: Cloud Alibabab笔记问世,全网详解仅此一份手慢无 什么是Spring cloud alibaba Spring Cloud Alibaba 是阿里巴巴提供的微服务开发一站式解决方案, ...
- Mybatis系列全解(七):全息视角看Dao层两种实现方式之传统方式与代理方式
封面:洛小汐 作者:潘潘 一直以来 他们都说为了生活 便追求所谓成功 顶级薪水.名牌包包 还有学区房 · 不过 总有人丢了生活 仍一无所获 · 我比较随遇而安 有些事懒得明白 平日里问心无愧 感兴趣的 ...
- 在windows 下查看ip 地址和 在ubundu 下查看IP地址
在windows 下查看ip 地址和 在ubundu 下查看IP地址 1.在windows 下查看 IP地址:ipconfig 2.在 ubundu 下查看IP地址:ifconfig
- 使用jsoup十分钟内掌握爬虫技术
对,就是十分钟,没有接触过爬虫的你,肯定一脸懵逼,感觉好高深的样子,一开始我也有点懵,但用了以后发现还是很简单的,java爬虫框架有很多,让我有种选择困难症,通过权衡比较还是感觉jsoup比较好用些, ...
- 「CTSC 2013」组合子逻辑
Tag 堆,贪心 Description 给出一个数列 \(n\) 个数,一开始有一个括号包含 \([1,n]\),你需要加一些括号,使得每个括号(包括一开始的)所包含的元素个数 \(\leq\) 这 ...
- .zip爆破
.zip爆破 Python的优化问题 Python在计算密集型任务方面没有明显的多线程优化,多线程更加适合用于处理I/O密集型任务(如网络请求).爆破任务使用顺序执行即可. 编写Python脚本 一个 ...
- java 实现快速排序
package leetcode;public class Singleton { public static void sort(int[] array,int low,int high){ int ...
- 《数据持久化与鸿蒙的分布式数据管理能力》直播课答疑和PPT分享
问:hi3861开发板支持分布式数据库吗? 目前,分布式数据库仅支持Java接口,因此Hi3861没有现成的API用于操作分布式数据库. 问:分布式数据管理包括搜索吗? 分布式数据管理包括融合搜索能力 ...
- IPFS挖矿的成本有哪些?
IPFS作为区块链新贵,近来风头一时无量.截止3月9日,Filecoin以257亿的流通市值超越门罗币,稳居区块链流通排行榜. 无论什么投资,其门槛一定在成本.今天就和大家细说投资市面上常见实体矿机的 ...
- 【小白学算法】5.链表(linked list)、链表的添加
链表其实也就是 线性表的链式存储结构,与之前讲到的顺序存储结构不同. 我们知道顺序存储结构中的元素地址都是连续的,那么这就有一个最大的缺点:当做插入跟删除操作的时候,大量的元素需要移动. 如图所示,元 ...