2015-03-30 2 views
0

У меня есть следующий код:рекурсия каталоги идет только один файл глубокого

find_info(File) -> 
    case file:read_file_info(File) of 
     {ok, Facts} -> 
      case Facts#file_info.type of 
       directory -> directory; 
       regular -> regular 
      end; 
     {error,Reason} -> exit(Reason) 
    end. 

find_files(Dir,Flag,Ending,Acc) -> 
    case file:list_dir(Dir) of 
     {ok,A} -> find_files_helper(A,Dir,Flag,Acc,Ending); 
     {_,_} -> Acc 
    end. 

find_files_helper([H|Tail],Dir,Flag,Acc,Ending) -> 
    A = find_info(filename:absname_join(Dir,H)), 
    case A of 
     directory -> 
      case Flag of 
       true -> 
        find_files(filename:absname_join(Dir,H),Flag,Ending,AcC++ find_files_helper(Tail,Dir,Flag,Acc,Ending)); 
       false -> find_files_helper(Tail,Dir,Flag,Acc,Ending) 
      end; 
     regular -> 
     case filename:extension(H) of 
      Ending -> find_files_helper(Tail,Dir,Flag,[to_md5_large(H)] ++ Acc, Ending); 
      _ -> find_files_helper(Tail,Dir,Flag,Acc,Ending) 
     end; 
     {error,Reason} -> exit(Reason) 
    end; 
find_files_helper([],_,_,Acc,_) -> Acc. 

Однако всякий раз, когда я бег find_files/4 программы только идет один файл глубоко, прежде чем врезаться. Скажем, у меня есть следующий каталог

home/ 
    a/ 
    ser.erl 
    b/ 
    c/ 
file.erl 
file2.erl 

При запуске я получить md5 от file.erl из file2.erl и из ser.erl. Однако, если каталог выглядит так:

home/ 
    a/ 
    ser.erl 
    back.erl 
    b/ 
    c/ 
file.erl 
file2.erl 

Тогда вся программа вылетает. Я потратил несколько хороших часов на то, что мне не хватает в моей логике, но я понятия не имею.

Сообщение об ошибке, которое я получаю, является исключительным enoent в функции p: to_md5_large/1.

В случае md5 необходимо здесь:

to_md5_large(File) -> 

     case file:read_file(File) of 
      {ok, <<B/binary>>} -> md5_helper(B,erlang:md5_init()); 
      {error,Reason} -> exit(Reason) 
     end. 

md5_helper(<<A:4/binary,B>>,Acc) -> md5_helper(B,erlang:md5_update(Acc,A)); 
md5_helper(A,Acc) -> 
    B =  erlang:md5_update(Acc,A), 
    erlang:md5_final(B). 
+0

Вы получаете 'enoent', потому что вы передаете имя файла, например' back.erl', 'to_md5_large', когда вы не находитесь в каталоге, где находится' back.erl'. Попробуйте передать полное имя файла. –

+0

@SteveVinoski хотят сказать это как ответ? – Bula

ответ

1

вы получаете enoent, потому что вы передаете имя файла, как back.erl к to_md5_large, когда вы не находитесь в директории, где находится back.erl. Попробуйте передать полное имя файла. Вы уже звоните filename:absname_join(Dir,H) в find_files_helper, поэтому просто сохраните это значение переменной и передайте эту переменную вместо H в to_md5_large.

1

Существует функция, которая делает это для вас:

fold_files(Dir, RegExp, Recursive, Fun, AccIn) -> AccOut 

в вашем случае:

Result = filelib:fold_files(Dir, ".*\.erl", true, fun(X,Acc) -> {ok,B} = file:read_file(X), [erlang:md5(B)|Acc] end, []). 

[править]

@Bula:

Я не ответил прямо на ваш вопрос по 2 причинам:

  • Первый заключается в том, в то время я писал свой ответ, вы не предоставили тип ошибки, которую вы получаете. Очень важно, с любого языка, узнать, как получить информацию из отчета об ошибке. В erlang, в большинстве случаев, вы получаете тип ошибки - это линия, где она встречается, глядя на документацию, у вас будет очень полезная информация о том, что происходит не так. Кстати, если вы не хотите, чтобы управлять ошибками, я отговорить вас писать такие вещи, как:

    case file:read_file(File) of 
        {ok, <<B/binary>>} -> md5_helper(B,erlang:md5_init()); 
        {error,Reason} -> exit(Reason) 
    end. 
    

Следующий код будет делать то же, короче, и вы получите точное число линий, где вам есть вопрос (его не самый лучший пример в коде, но это более короткий)

{ok, <<B/binary>>} = file:read_file(File), 
    md5_helper(B,erlang:md5_init()), 
  • во-вторых, я считаю, что ваш код слишком большой, с ненужными вспомогательных функций. Я думаю, что важно попытаться получить сжатый и читаемый код, а также попытаться правильно использовать библиотечную функцию. Например, вы используете erlang: md5: init/0, erlang: md5_update/2 и erlang: md5_final/1, в то время как в вашем случае достаточно одного вызова erlang: md5/1. Способ, которым вы его используете, существует, чтобы иметь возможность вычислить md5, когда вы получаете кусок данных куском, что не является вашим делом, и способ, которым вы написали вспомогательную функцию, не позволяет использовать эту функцию.

Я не понимаю, почему вы хотите иметь «развернутую» версию своего кода, но я предлагаю вам другую версию, где я пытался следовать моим советам (написанные непосредственно в оболочке, поэтому для этого требуется R17 + для определение рекурсивной анонимной функции): о)

1> F = fun F(X,D,Ending) ->                  
1> {ok,StartD} = file:get_cwd(),  %% save current directory          
1> ok = file:set_cwd(D),    %% move to the directory to explore       
1> R = case filelib:is_dir(X) of                 
1>  true ->       %% if the element to analyze is a directory         
1>  {ok,Files} = file:list_dir(X), %% getits content          
1>  [F(Y,X,Ending) || Y <- Files]; %% and recursively analyze all its elements    
1>  false -> 
1>  case filelib:is_regular(X) andalso (filename:extension(X) == Ending) of   
1>   true ->      %% if it is a regular file with the right extension    
1>   {ok,B} = file:read_file(X), %% read it            
1>   [erlang:md5(B)];    %% and calculate the md5 (must be return in a list            
1>          %% for consistancy with directory results) 
1>   false ->                     
1>   []       %% in other cases (symlink, ...) return empty          
1>  end                      
1> end,                       
1> ok = file:set_cwd(StartD),   %% restore current directory          
1> lists:flatten(R)      %% flatten for nicer result          
1> end.                       
#Fun<erl_eval.42.90072148> 
2> Md5 = fun(D) -> F(D,D,".erl") end. 
#Fun<erl_eval.6.90072148> 
3> Md5("C:/My programs/erl6.2/lib/stdlib-2.2"). 
[<<150,238,21,49,189,164,184,32,42,239,200,52,135,78,12, 
    112>>, 
<<226,53,12,102,125,107,137,149,116,47,50,30,37,13,211,243>>, 
<<193,114,120,24,175,27,23,218,7,169,146,8,19,208,73,255>>, 
<<227,219,237,12,103,218,175,238,194,103,52,180,132,113, 
    184,68>>, 
<<6,16,213,41,39,138,161,36,184,86,17,183,125,233,20,125>>, 
<<23,208,91,76,69,173,159,200,44,72,9,9,50,40,226,27>>, 
<<92,8,168,124,230,1,167,199,6,150,239,62,146,119,83,36>>, 
<<100,238,68,145,58,22,88,221,179,204,19,26,50,172,142,193>>, 
<<253,79,101,49,78,235,151,104,188,223,55,228,163,25,16, 
    147>>, 
<<243,189,25,98,170,97,88,90,174,178,162,19,249,141,94,60>>, 
<<237,85,6,153,218,60,23,104,162,112,65,69,148,90,15,240>>, 
<<225,48,238,193,120,43,124,63,156,207,11,4,254,96,250,204>>, 
<<67,254,107,82,106,87,36,119,140,78,216,142,66,225,8,40>>, 
<<185,246,227,162,211,133,212,10,174,21,204,75,128,125, 
    200,...>>, 
<<234,191,210,59,62,148,130,187,60,0,187,124,150,213,...>>, 
<<199,231,45,34,185,9,231,162,187,130,134,246,54,...>>, 
<<157,226,127,87,191,151,81,50,19,116,96,121,...>>, 
<<15,59,143,114,184,207,96,164,155,44,238,...>>, 
<<176,139,190,30,114,248,0,144,201,14,...>>, 
<<169,79,218,157,20,10,20,146,12,...>>, 
<<131,25,76,110,14,183,5,103,...>>, 
<<91,197,189,2,48,142,67,...>>, 
<<94,202,72,164,129,237,...>>, 
<<"^NQÙ¡8hÿèkàå"...>>,<<"ðÙ.Q"...>>, 
<<150,101,76,...>>, 
<<"A^ÏrÔ"...>>,<<"¹"...>>,<<...>>|...] 
4> 
+0

Я не хочу использовать BIF – Bula

+1

Нет там BIF? и это странное требование. – Pascal