问题描述
这个警告有多严重?我可以在出现此警告的情况下使用此模块吗?这个警告的副作用是什么。
警告:未定义的回调函数终止/3(行为 'gen_statem')**
Erlang/OTP: 19.0.7
git clone https://github.com/inaka/apns4erl.git
cd apns4erl/
wget https://s3.amazonaws.com/rebar3/rebar3 && chmod +x rebar3
./rebar3 compile
===> Fetching coveralls v2.2.0
===> Fetching jsx v2.10.0
===> Analyzing applications...
===> Compiling jsx
===> Compiling coveralls
===> Verifying dependencies...
===> Fetching base64url v1.0.1
===> Fetching gun v1.3.3
===> Fetching jsx v3.0.0
===> Fetching cowlib v2.7.3
===> Analyzing applications...
===> Compiling cowlib
===> Compiling base64url
===> Compiling gun
===> Compiling jsx
===> Analyzing applications...
===> Compiling apns
**src/apns_connection.erl:22: Warning: undefined callback function terminate/3 (behavIoUr 'gen_statem')**
解决方法
这个警告有多严重?
应该定义的函数不存在。
我可以在出现此警告的情况下使用此模块吗?
Supervisors 使用 terminate() 函数在必要时关闭子进程。如果子进程永远不需要关闭,那么你很好。
你可以在源代码中添加一个什么都不做的终止函数:
terminate(_Reason,_State,_Data) -> ok.
但是,更好的选择可能是将错误报告给 github 存储库,希望他们能够快速修复代码。
或者,您可以尝试升级您的 erlang 版本。在较新的 erlang 版本中,会自动为您定义 terminate() 函数。
这个警告的副作用是什么。
您的程序有时会崩溃。
,在 Erlang/OTP 19.3 及更高版本中,由于 this change,terminate/3
回调对于 gen_statem 是可选的。如the documentation中所述:
这个回调是可选的,所以回调模块不需要导出它。 gen_statem 模块提供了一个无需清理的默认实现。
那么让我们弄清楚在早期版本中不包含此功能的后果是什么。根据文档,terminate
函数将在三种不同的情况下被调用:
- 另一个回调函数在
{stop,Reason}
中返回了一个停止元组Actions
- gen_statem 是监督树的一部分,并由其监督者命令终止(仅当 gen_statem 已设置为捕获退出信号且关闭策略不是
brutal_kill
) - 进程收到来自其父进程的
'EXIT'
消息
当然可以想象,对于特定的 gen_statem 不会出现这些情况,因此缺少 terminate
函数不会产生负面影响。在最坏的情况下,terminate
函数会在进程即将退出时被调用,从而以与原始退出原因不同的退出原因终止进程,从而隐藏了原始问题的原因。