Erlang-재귀

재귀는 Erlang의 중요한 부분입니다. 먼저 팩토리얼 프로그램을 구현하여 간단한 재귀를 구현하는 방법을 살펴 보겠습니다.

-module(helloworld). 
-export([fac/1,start/0]). 

fac(N) when N == 0 -> 1; 
fac(N) when N > 0 -> N*fac(N-1). 

start() -> 
   X = fac(4), 
   io:fwrite("~w",[X]).

위 프로그램에 대해 다음 사항에 유의해야합니다.

  • 먼저 fac (N)이라는 함수를 정의합니다.

  • fac (N)을 재귀 적으로 호출하여 재귀 함수를 정의 할 수 있습니다.

위 프로그램의 출력은 다음과 같습니다.

산출

24

재귀에 대한 실용적인 접근 방식

이 섹션에서는 다양한 유형의 재귀와 Erlang에서의 사용법에 대해 자세히 설명합니다.

길이 재귀

재귀에 대한보다 실용적인 접근 방식은 목록의 길이를 결정하는 데 사용되는 간단한 예를 통해 볼 수 있습니다. 목록에는 [1,2,3,4]와 같은 여러 값이있을 수 있습니다. 재귀를 사용하여 목록의 길이를 얻는 방법을 살펴 보겠습니다.

Example

-module(helloworld). 
-export([len/1,start/0]). 

len([]) -> 0; 
len([_|T]) -> 1 + len(T). 

start() -> 
   X = [1,2,3,4], 
   Y = len(X), 
   io:fwrite("~w",[Y]).

위 프로그램에 대해 다음 사항에 유의해야합니다.

  • 첫 번째 기능 len([]) 목록이 비어있는 경우 특수한 경우 조건에 사용됩니다.

  • 그만큼 [H|T] 하나 이상의 요소 목록과 일치하는 패턴입니다. 길이 목록은 다음과 같이 정의됩니다. [X|[]] 길이 2의 목록은 다음과 같이 정의됩니다. [X|[Y|[]]]. 두 번째 요소는 목록 자체입니다. 즉, 첫 번째 요소 만 계산하면되고 함수는 두 번째 요소에서 자신을 호출 할 수 있습니다. 목록의 각 값은 길이 1로 계산됩니다.

위 프로그램의 출력은 다음과 같습니다.

Output

4

꼬리 재귀

꼬리 재귀가 어떻게 작동하는지 이해하기 위해 이전 섹션의 다음 코드가 어떻게 작동하는지 이해합시다.

Syntax

len([]) -> 0; 
len([_|T]) -> 1 + len(T).

1 + len (Rest)에 대한 답은 len (Rest)의 답을 찾아야합니다. len (Rest) 함수 자체는 다른 함수 호출의 결과를 찾아야했습니다. 추가 된 항목은 마지막 항목이 발견 될 때까지 누적 된 다음 최종 결과가 계산됩니다.

테일 재귀는 발생하는 작업을 줄임으로써 이러한 작업 스택을 제거하는 것을 목표로합니다.

이를 위해 함수의 매개 변수로 추가 임시 변수를 보유해야합니다. 앞서 언급 한 임시 변수는 때때로 누산기라고 불리며 호출의 증가를 제한하기 위해 발생하는 계산 결과를 저장하는 장소 역할을합니다.

꼬리 재귀의 예를 살펴 보겠습니다.

Example

-module(helloworld).
-export([tail_len/1,tail_len/2,start/0]). 

tail_len(L) -> tail_len(L,0). 
tail_len([], Acc) -> Acc; 
tail_len([_|T], Acc) -> tail_len(T,Acc+1). 

start() -> 
   X = [1,2,3,4], 
   Y = tail_len(X), 
   io:fwrite("~w",[Y]).

위 프로그램의 출력은 다음과 같습니다.

Output

4

복제

재귀의 예를 살펴 보겠습니다. 이번에는 정수를 첫 번째 매개 변수로 사용하고 다른 용어를 두 번째 매개 변수로 사용하는 함수를 작성해 보겠습니다. 그런 다음 정수로 지정된 용어의 복사본 목록을 만듭니다.

이 예제가 어떻게 생겼는지 살펴 보겠습니다.

-module(helloworld). 
-export([duplicate/2,start/0]). 

duplicate(0,_) -> 
   []; 
duplicate(N,Term) when N > 0 ->
   io:fwrite("~w,~n",[Term]),
   [Term|duplicate(N-1,Term)]. 
start() -> 
   duplicate(5,1).

위 프로그램의 출력은 다음과 같습니다.

산출

1,
1,
1,
1,
1,

목록 반전

Erlang에서 재귀를 사용할 수있는 경계는 없습니다. 이제 재귀를 사용하여 목록의 요소를 반전 할 수있는 방법을 빠르게 살펴 보겠습니다. 이를 위해 다음 프로그램을 사용할 수 있습니다.

-module(helloworld). 
-export([tail_reverse/2,start/0]). 

tail_reverse(L) -> tail_reverse(L,[]).

tail_reverse([],Acc) -> Acc; 
tail_reverse([H|T],Acc) -> tail_reverse(T, [H|Acc]).

start() -> 
   X = [1,2,3,4], 
   Y = tail_reverse(X), 
   io:fwrite("~w",[Y]).

위 프로그램의 출력은 다음과 같습니다.

산출

[4,3,2,1]

위 프로그램에 대해 다음 사항에 유의해야합니다.

  • 다시 임시 변수의 개념을 사용하여 목록의 각 요소를 Acc라는 변수에 저장합니다.

  • 그런 다음 전화 tail_reverse 재귀 적으로하지만 이번에는 마지막 요소가 새 목록에 먼저 배치되도록합니다.

  • 그런 다음 목록의 각 요소에 대해 tail_reverse를 재귀 적으로 호출합니다.