当我想编译一个简单的file.erl
文件时,我收到了错误消息。
$ erlc file.erl
/opt/erlang17.5/lib/kernel-3.2/include/file.hrl:34: type date_time() undefined
以下是file.erl的内容
-module(file).
-include_lib("kernel/include/file.hrl").
-export([file_info/1]).
file_info(Dir) ->
{ok, F} = file:read_file_info(Dir),
io:format("~p~n", [F#file_info.type]).
似乎没有人报告过这种问题,我做了什么有什么问题?
答案 0 :(得分:6)
调查编译问题的根本原因是有益的。编译器发出此错误:
/opt/erlang17.5/lib/kernel-3.2/include/file.hrl:34:输入date_time()undefined
查看file.hrl
的第34行,我们看到:
ctime :: file:date_time() | non_neg_integer(),
这是ctime
记录的#file_info{}
字段的类型说明符。它声明ctime
可以包含file:date_time()
类型或non_neg_integer()
类型的值。
错误消息特别抱怨date_time()
类型,因为其范围表明它已在file
模块中定义。在这种情况下,file
模块是正在定义和编译的模块,它隐藏了Erlang / OTP kernel
应用程序提供的标准模块。标准file
模块定义其date_time()
类型,如下所示:
-type date_time() :: calendar:datetime().
我们尝试编译的file
模块没有。如果我们添加此类型定义会发生什么?
-module(file).
-type date_time() :: calendar:datetime().
-include_lib("kernel/include/file.hrl").
-export([file_info/1]).
file_info(Dir) ->
{ok, F} = file:read_file_info(Dir),
io:format("~p~n", [F#file_info.type]).
现在可以编译包含添加的类型定义的此版本:
$ erlc file.erl
$
我们修复了编译错误,所以让我们尝试使用我们编译的模块。我们启动一个Erlang shell:
$ erl
Erlang/OTP 18 [erts-7.0] [source] [64-bit] [smp:8:8] [ds:8:8:10] [async-threads:10] [hipe] [kernel-poll:false]
{"init terminating in do_boot",{undef,[{file,path_eval,[[".","/Users/user"],".erlang"],[]},{c,f_p_e,2,[{file,"c.erl"},{line,481}]},{init,eval_script,8,[]},{init,do_boot,3,[]}]}}
Crash dump is being written to: erl_crash.dump...done
init terminating in do_boot ()
我们可以在错误消息中看到尝试调用file:path_eval/2
失败,因为它未定义。由于我们的file
模块隐藏了标准模块,但没有提供所有相同的功能,因此只要我们的file
模块隐藏标准模块,我们甚至无法启动Erlang shell
显然,为模块选择不同的名称不仅仅是修复原始编译错误。
答案 1 :(得分:1)
您的代码是正确的,问题在于文件名。 Erlang有库"文件",所以你不能使用这个文件名。改变它(不要忘记修改源代码),一切都应该没问题。