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를 재귀 적으로 호출합니다.