MIX OTP——监督树和应用

在上一章关于 GenServer 的内容中,我们实现了 KV.Registry 来管理存储容器。在某个时候,我们开始监控存储容器,这样每当 KV.Bucket 崩溃时,我们就能采取行动。虽然变化相对较小,但它提出了一个 Elixir 开发人员经常问的问题:当出现故障时会发生什么?

在我们添加监控之前,如果存储容器崩溃,注册表将永远指向不再存在的存储容器。如果用户尝试读取或写入崩溃的存储容器,它将失败。任何尝试创建具有相同名称的新存储容器的操作都只会返回崩溃存储容器的 PID。换句话说,该存储容器的注册表条目将永远处于不良状态。一旦我们添加了监控,注册表就会自动删除崩溃存储容器的条目。现在尝试查找崩溃的存储容器(正确)会显示存储容器不存在,系统用户可以根据需要成功创建一个新的存储容器。

实际上,我们并不希望作为存储容器工作的进程失败。但是,如果确实发生了这种情况,无论出于何种原因,我们都可以放心,我们的系统将继续按预期工作。

如果您有编程经验,您可能会想:“我们能保证存储容器不会首先崩溃吗?”。正如我们将看到的,Elixir 开发人员倾向于将这些做法称为“防御性编程”。这是因为实时生产系统有几十种不同的原因导致某些事情可能出错。磁盘可能会发生故障,内存可能会损坏,错误,网络可能会停止工作一秒钟,等等。如果我们要编写试图保护或规避所有这些错误的软件,我们将花费更多时间来处理故障,而不是编写自己的软件!

因此,Elixir 开发人员更喜欢“让它崩溃”或“快速失败”。我们从故障中恢复的最常见方法之一是重新启动系统崩溃的任何部分。

例如,想象一下您的计算机、路由器、打印机或任何设备无法正常工作。您多久通过重新启动来修复它一次?一旦我们重新启动设备,我们就会将设备重置回其初始状态,该状态经过充分测试并保证可以正常工作。在 Elixir 中,我们将同样的方法应用于软件:每当一个进程崩溃时,我们都会启动一个新进程来执行与崩溃进程相同的工作。

在 Elixir 中,这是由 Supervisor 完成的。Supervisor 是一个监督其他进程并在它们崩溃时重新启动它们的进程。为此,Supervisor 管理任何受监督进程的整个生命周期,包括启动和关闭。

在本章中,我们将学习如何通过监督 KV.Registry 进程将这些概念付诸实践。毕竟,如果注册表出现问题,整个注册表都会丢失,并且永远找不到任何存储容器!为了解决这个问题,我们将定义一个 KV.Supervisor 模块,以确保我们的 KV.Registry 在任何给定时刻都处于启动和运行状态。

在本章的最后,我们还将讨论应用程序。正如我们将看到的,Mix 已将我们所有的代码打包到一个应用程序中,我们将学习如何定制我们的应用程序,以确保我们的 Supervisor 和 Registry 在系统启动时正常运行。

我们的第一个监督者

监督者是一个监督其他进程的进程,我们将其称为子进程。监督进程的行为包括三个不同的职责。第一个是启动子进程。一旦子进程运行因为异常终止或达到某个条件,监督者可能会重新启动子进程。例如,如果任何子进程死亡,监督者可能会重新启动所有子进程。最后,监督者还负责在系统关闭时关闭子进程。有关更深入的讨论,请参阅监督者模块。

创建监督者与创建 GenServer 没有太大区别。我们将在 lib/kv/supervisor.ex 文件中定义一个名为 KV.Supervisor 的模块,它将使用 Supervisor 行为:

到目前为止,我们的监督者只有一个子进程:KV.Registry。定义子进程列表后,我们调用 Supervisor.init/2,传递子进程和监督策略。

监督策略规定当其中一个子进程崩溃时会发生什么。:one_for_one 表示如果子进程死亡,它将是唯一重新启动的进程。由于我们现在只有一个子进程,这就是我们所需要的。监督者行为支持多种策略,我们将在本章中讨论。

一旦监督者启动,它将遍历子进程列表,并在每个模块上调用 child_spec/1 函数。

child_spec/1 函数返回子进程规范,该规范描述了如何启动进程,进程是工作者进程还是监督者进程,进程是临时的、瞬态的还是永久的等等。当我们使用 Agent、使用 GenServer、使用 Supervisor 等时,会自动定义 child_spec/1 函数。让我们在终端中使用 iex -S mix 尝试一下:

随着我们继续学习本指南,我们将了解这些细节。如果您想提前了解,请查看 Supervisor 文档。

在 Supervisor 检索所有子规范后,它会按照子规范中 :start 键中的信息,按照定义顺序逐个启动其子规范。对于我们当前的规范,它将调用 KV.Registry.start_link([])。

让我们试用一下 Supervisor:

到目前为止,我们已经启动了监督程序并列出了其子程序。一旦监督程序启动,它也会启动其所有子程序。

如果我们故意使监督程序启动的注册表崩溃,会发生什么?让我们通过在调用时向其发送错误输入来实现这一点:

请注意,一旦我们因错误输入导致注册表崩溃,监管者就会自动启动一个具有新 PID 的新注册表来代替第一个注册表。

在前面的章节中,我们总是直接启动进程。例如,我们将调用 KV.Registry.start_link([]),它将返回 {:ok, pid},这将允许我们通过其 pid 与注册表进行交互。既然进程是由监管者启动的,我们必须直接询问监管者它的子进程是谁,并从返回的子进程列表中获取 PID。实际上,每次这样做都会非常昂贵。为了解决这个问题,我们经常给进程命名,允许它们在我们的代码中的任何位置在单个机器中被唯一地标识。

让我们学习如何做到这一点。

命名进程

虽然我们的应用程序有许多存储容器,但它只有一个注册表。因此,每当我们启动注册表时,我们都希望为其赋予一个唯一的名称,以便我们可以从任何地方访问它。我们通过将 :name 选项传递给 KV.Registry.start_link/1 来实现这一点。

让我们稍微改变一下我们的 children 定义(在 KV.Supervisor.init/1 中),将其改为元组列表,而不是原子列表:

有了这个,监督者现在将通过调用 KV.Registry.start_link(name: KV.Registry) 来启动 KV.Registry。

如果您重新访问 KV.Registry.start_link/1 实现,您会记得它只是将选项传递给 GenServer:

反过来,它将使用给定的名称注册进程。:name 选项需要一个用于本地命名进程的原子(本地命名意味着它可用于此机器 - 还有其他选项,我们不会在这里讨论)。由于模块标识符是原子(在 IEx 中尝试 i(KV.Registry)),我们可以用实现它的模块来命名进程,前提是该名称只有一个进程。这有助于调试和自省系统。

让我们在 iex -S mix 中尝试更新后的监督器:

这次,监管者启动了一个命名注册表,这样我们就可以创建存储容器,而不必从监管者那里显式获取 PID。您还应该知道如何使注册表再次崩溃,而无需查找其 PID:试一试。

此时,您可能想知道:您还应该在本地命名存储容器进程吗?请记住,存储容器是根据用户输入动态启动的。由于本地名称必须是原子,因此我们必须动态创建原子,这是一个坏主意,因为一旦定义了原子,它就永远不会被擦除或垃圾收集。这意味着,如果我们根据用户输入动态创建原子,我们最终将耗尽内存(或者更准确地说,VM 将崩溃,因为它对原子数量施加了硬性限制)。这个限制正是我们创建自己的注册表的原因(或者为什么人们会使用 Elixir 的内置注册表模块)。

我们越来越接近一个完全正常工作的系统。监管者会自动启动注册表。但是,我们如何在系统启动时自动启动监管者?要回答这个问题,让我们谈谈应用程序。

了解应用程序

我们一直在应用程序内部工作。每次我们更改文件并运行 mix compile 时,我们都可以在编译输出中看到一条 Generated kv app 消息。

我们可以在 _build/dev/lib/kv/ebin/kv.app 找到生成的 .app 文件。让我们看看它的内容:

此文件包含 Erlang 术语(使用 Erlang 语法编写)。即使我们不熟悉 Erlang,也很容易猜到这个文件包含我们的应用程序定义。它包含我们的应用程序版本、它定义的所有模块,以及我们依赖的应用程序列表,如 Erlang 的内核、elixir 本身和记录器。

记录器应用程序作为 Elixir 的一部分提供。我们通过在 mix.exs 中的 :extra_applications 列表中指定它来表明我们的应用程序需要它。有关更多信息,请参阅官方文档。

简而言之,应用程序由 .app 文件中定义的所有模块组成,包括 .app 文件本身。应用程序通常只有两个目录:ebin,用于存放 Elixir 工件,例如 .beam 和 .app 文件;priv,用于存放应用程序中可能需要的任何其他工件或资产。

虽然 Mix 会为我们生成并维护 .app 文件,但我们可以通过在 mix.exs 项目文件内的 application/0 函数中添加新条目来自定义其内容。我们很快就会进行第一次自定义。

启动应用程序

我们系统中的每个应用程序都可以启动和停止。启动和停止应用程序的规则也在 .app 文件中定义。当我们调用 iex -S mix 时,Mix 会编译我们的应用程序然后启动它。

让我们在实践中看看这一点。使用 iex -S mix 启动控制台并尝试:

它已经启动了。Mix 会自动启动当前应用程序及其所有依赖项。对于 mix test 和许多其他 Mix 命令也是如此。

但是,我们可以停止 :kv 应用程序以及 :logger 应用程序:

让我们尝试再次启动我们的应用程序:

现在我们收到错误,因为 :kv 所依赖的应用程序(在本例中为 :logger)未启动。我们需要以正确的顺序手动启动每个应用程序,或者调用 Application.ensure_all_started/1,如下所示:

实际上,我们的工具总是会为我们启动应用程序,但如果您需要细粒度的控制,可以使用 API。

应用程序回调

每当我们调用 iex -S mix 时,Mix 都会通过调用 Application.start(:kv) 自动启动我们的应用程序。但是我们可以自定义应用程序启动时发生的情况吗?事实上,我们可以!为此,我们定义一个应用程序回调。

第一步是告诉我们的应用程序定义(例如,我们的 .app 文件)哪个模块将实现应用程序回调。让我们通过打开 mix.exs 并将 def application 更改为以下内容来做到这一点:

:mod 选项指定“应用程序回调模块”,后跟应用程序启动时要传递的参数。应用程序回调模块可以是实现应用程序行为的任何模块。

要实现应用程序行为,我们必须使用 Application 并定义一个 start/2 函数。start/2 的目标是启动一个监督器,然后它将启动任何子服务或执行我们的应用程序可能需要的任何其他代码。让我们利用这个机会启动我们在本章前面实现的 KV.Supervisor。

由于我们已将 KV 指定为模块回调,因此让我们更改 lib/kv.ex 中定义的 KV 模块以实现 start/2 函数:

请注意,这样做会破坏测试 KV 中 hello 函数的样板测试用例。您可以简单地删除该测试用例。

当我们使用 Application 时,我们可能会定义几个函数,类似于使用 Supervisor 或 GenServer 时。这次我们只需要定义一个 start/2 函数。Application 行为也有一个 stop/1 回调,但在实践中很少使用。您可以查看文档以获取更多信息。

现在您已经定义了一个启动我们的监督器的应用程序回调,我们希望 KV.Registry 进程在我们启动 iex -S mix 时立即启动并运行。让我们再试一次:

让我们回顾一下发生了什么。每当我们调用 iex -S mix 时,它都会通过调用 Application.start(:kv) 自动启动我们的应用程序,然后调用应用程序回调。应用程序回调的工作是启动监督树。目前,我们的监督者有一个名为 KV.Registry 的子节点,以名称 KV.Registry 开头。我们的监督者可以有其他子节点,其中一些子节点可以成为自己的监督者,并拥有自己的子节点,从而形成所谓的监督树。

项目还是应用程序?

Mix 区分了项目和应用程序。根据我们的 mix.exs 文件的内容,我们可以说我们有一个定义 :kv 应用程序的 Mix 项目。正如我们将在后面的章节中看到的那样,有些项目没有定义任何应用程序。

当我们说“项目”时,您应该考虑 Mix。Mix 是管理项目的工具。它知道如何编译您的项目、测试您的项目等等。它还知道如何编译和启动与您的项目相关的应用程序。

当我们谈论应用程序时,我们谈论的是 OTP。应用程序是由运行时作为一个整体启动和停止的实体。您可以在应用程序模块的文档中了解有关应用程序的更多信息以及它们与整个系统的启动和关闭的关系。

下一步

虽然本章是我们第一次实现监督器,但这并不是我们第一次使用它!在上一章中,当我们在测试期间使用 start_supervised! 启动注册表时,ExUnit 在由 ExUnit 框架本身管理的监督器下启动了注册表。通过定义我们自己的监督器,我们提供了更多关于如何在应用程序中初始化、关闭和监督进程的结构,使我们的生产代码和测试与最佳实践保持一致。

但我们还没有完成。到目前为止,我们正在监督注册表,但我们的应用程序也在启动存储容器。由于存储容器是动态启动的,我们可以使用一种称为 DynamicSupervisor 的特殊类型的监督器,它经过优化以处理此类场景。接下来让我们探索它。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/863343.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

独家原创 | Matlab实现CNN-Transformer多变量时间序列预测

SCI一区级 | Matlab实现BO-Transformer-GRU多变量时间序列预测 目录 SCI一区级 | Matlab实现BO-Transformer-GRU多变量时间序列预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现CNN-Transformer多变量时间序列预测; 2.运行环境为Matlab2023b…

【JavaScript】JavaScript简介

希望文章能给到你启发和灵感~ 如果觉得文章对你有帮助的话,点赞 关注 收藏 支持一下博主吧~ 阅读指南 JavaScript入门(1)————JavaScript简介开篇说明一、什么是JavaScript二、JavaScript的使用2.1 开发工具的选择…

fiddler抓包工具

概念 概念: Fiddler是一个http协议调试代理工具,它能够记录并检查所有你的电脑和互联网之间的http通讯。 http:不加密,端口为80 https:加密,端口为443 原理: 其实就在访问服务器时&#xff0…

如何在写代码中找到乐趣

平时我们写代码呢,多数情况都是流水线式写代码,基本就可以实现业务逻辑了。 如何在写代码中找到乐趣呢,我觉得,最好的方式就是:使用设计模式优化自己的业务代码。 参考资料: 实战!工作中常用到…

[方法] Unity 3D模型与骨骼动画

1. 在软件中导出3D模型 1.1 3dsmax 2014 1.1.1 TGA转PNG 3dsmax的贴图格式为tga,我们需要在在线格式转换中将其转换为Unity可识别的png格式。 1.1.2 模型导出 导出文件格式为fbx。在导出设置中,要勾选三角算法,取消勾选摄像机和灯光&#…

三秒4张图!让 Stable Diffusion 出图速度暴增的新一代生成模型LCM!

前言 大家好,这里是和你们一起探索 AI绘画月月~ 最近一种新的图像生成形式逐渐兴起,即生成的图像会随输入的文字或笔画动作迅速变化,这让图像生成有了更多灵活探索和准确控制的空间。这种「实时反馈」的感觉源于模型能在几秒钟内&#xff0…

fiddler 返回Raw乱码

有时会发现自己发送的请求后,返回结果Raw里面是乱码,可以勾选Decode并重新发送请求就解决了 这个时候将Decode勾选一下 此时就好了

【C++ | 委托构造函数】委托构造函数 详解 及 例子源码

😁博客主页😁:🚀https://blog.csdn.net/wkd_007🚀 🤑博客内容🤑:🍭嵌入式开发、Linux、C语言、C、数据结构、音视频🍭 🤣本文内容🤣&a…

模版总结小全

BFS 最短步数问题 #include<iostream> #include<queue> #include<cstring> using namespace std;const int N 50; char g[N][N],d[N][N]; int dx[] {-1,0,1,0}; int dy[] {0,1,0,-1}; int n,m;int bfs(int x,int y){queue<pair<int,int> > q…

MySQL高级-SQL优化-insert优化-批量插入-手动提交事务-主键顺序插入

文章目录 1、批量插入1.1、大批量插入数据1.2、启动Linux中的mysql服务1.3、客户端连接到mysql数据库&#xff0c;加上参数 --local-infile1.4、查询当前会话中 local_infile 系统变量的值。1.5、开启从本地文件加载数据到服务器的功能1.6、创建表 tb_user 结构1.7、上传文件到…

mysql_config 命令, 可以查看mysqlclient库的位置在/usr/lib64/mysql下

好吧&#xff0c;其实我是从这里知道了 -l 后面加的库名和so文件这种名不一样&#xff0c;因为库文件实际叫下面这个名&#xff08;前面有lib)。

MySQL之覆盖索引

什么是覆盖索引&#xff1f; 覆盖索引&#xff1a;查询时使用了索引&#xff0c;且需要返回的列&#xff0c;在改索引中已经全部能找到。 示例&#xff1a;有user表如下&#xff1a; CREATE TABLE user (id bigint(20) NOT NULL AUTO_INCREMENT COMMENT 技术主键,name varch…

Git企业开发---初识Git

顾得泉&#xff1a;个人主页 个人专栏&#xff1a;《Linux操作系统》 《C从入门到精通》 《LeedCode刷题》 键盘敲烂&#xff0c;年薪百万&#xff01; 引言 不知道大家有没有经历这种困惑&#xff0c;当我们要去交某文档时&#xff0c;总是要进行修改&#xff0c;修改后再交…

Softmax函数的作用

Softmax 函数主要用于多类别分类问题&#xff0c;它将输入的数值转换为概率分布。 具体来说&#xff0c;对于给定的输入向量 x [x_1, x_2,..., x_n] &#xff0c;Softmax 函数的输出为 y [y_1, y_2,..., y_n] &#xff0c;其中&#xff1a; 这样&#xff0c;Softmax 函数的输…

人生最有力,最棒的十句话!

人生最有力&#xff0c;最棒的十句话 1、允许一切事发生&#xff0c;所有一切发生的事不是你能阻挡了的&#xff0c;你接受&#xff0c;他也发生&#xff0c;你不接受&#xff0c;他也发生&#xff0c;你还不如坦然面对接受现实。 2、你焦虑的时候千万不要躺着啥也不干&#xf…

全网唯一免费无水印AI视频工具!

最近Morph Studio开始免费公测&#xff01;支持高清画质&#xff0c;可以上传语音&#xff0c;同步口型&#xff0c;最重要的是生成的视频没有水印&#xff01; Morph Studio国内就可以访问&#xff0c;可以使用国内邮箱注册&#xff08;我用的163邮箱&#xff09;&#xff0c;…

Java--回顾方法的调用

1.静态方法与非静态方法 1.当二者皆为静态方式时&#xff0c;可直接类名.方法名调用其方法 2.当调用的方法是静态&#xff0c;被调用的方法为非静态时&#xff0c;调用将会报错 3.出现2情况可通过进行实例化这个类的方式进行调用&#xff0c;如图所示 4.当处于一个类下&#xf…

在IDEA中创建Maven项目

2023版IDEA创建Maven项目&#xff08;新版&#xff09; 1.打开IDEA&#xff0c;点击 文件 -> 新建 -> 项目 2.创建Maven项目 3.编写java文件并运行 在src -> java -> 创建一个java文件并运行 如果出现下图 解决办法&#xff1a; 2022版IDEA创建Maven项目&#xf…

判断时间序列中的元素是否为:年初、年末、季初、季末

【小白从小学Python、C、Java】 【考研初试复试毕业设计】 【Python基础AI数据分析】 判断时间序列中的元素是否为&#xff1a; 年初、年末、季初、季末 Series.dt.is_year_start Series.dt.is_year_end Series.dt.is_quarter_start Series.dt.is_quarter_end 选择题 关于以下…

J018_冒泡排序

一、排序过程 如果要对一个数组进行升序排序&#xff1a; 每个轮次两两数字进行比较&#xff0c;如果前面的数字大于后面的数字&#xff0c;则交换两个数字的位置&#xff1b;如果前面的数字小于或等于后面的数字&#xff0c;则这两个数字位置不变。直到把数组中所有数字比较…