> Java > java지도 시간 > 트램펄린, Java의 예

트램펄린, Java의 예

Barbara Streisand
풀어 주다: 2025-01-17 20:18:09
원래의
554명이 탐색했습니다.

Trampolim, exemplo em Java

n부터 0까지 숫자를 더하는 간단한 프로그램을 작성해 보겠습니다. 하지만 반복적 접근 방식 대신 재귀적 접근 방식을 사용해 보는 것은 어떨까요?

이 프로그램을 sum이라고 부릅니다. 우리는 sum(0) == 0을 알고 있으므로 이것이 기본 사례입니다. 기본 사례에 어떻게 도달합니까? sum(n) == n sum(n-1), 마침내 sum(0)에 도달할 때까지. 자바 코드는 다음과 같습니다.

<code class="language-java">int sum(int n) {
    if (n == 0) {
        return 0;
    }
    return n + sum(n - 1);
}</code>
로그인 후 복사
로그인 후 복사

재귀 문제가 있나요?

재귀에는 기본 사례가 입력 값에서 멀리 떨어져 있을 때 고유한 결함이 있습니다... 대부분의 언어에서 함수 호출은 프로그램의 스택을 사용하여 함수 호출 정보를 저장하므로 매우 큰 재귀로 인해 스택 오버플로가 발생할 수 있습니다.

그런데, 이를 피할 수 있는 방법이 있을까요? 실제로 그렇습니다. 이것은 트램폴린이라는 오래된 전략입니다.

스프링보드

스프링보드 전략의 기본 아이디어는 프로그램의 일부가 "값" 또는 "연속"을 반환한다는 것입니다. 계속이란 무엇입니까? 처리를 계속하는 함수입니다.

대략 다음과 같습니다.

<code class="language-java">let trampolim = primeiraChamada(input);

while (trampolim is continuation) {
    trampolim = trampolim.continue();
}
return trampolim;</code>
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사

sum의 연속은 무엇인가요?

프로그램을 다음과 같이 모델링sum하겠습니다. 단순히 반복하는 대신 연속을 사용합니다. 한 가지 방법은 acc을 연속을 통해 전달되는 객체로 사용하는 것입니다. 따라서 sum_trampoline(0, acc)에 도달하면 acc을 반환합니다. 진행하는 방법?

sum_trampoline(n, acc)에서 sum_trampoline(n-1, acc n)으로 가보겠습니다. 첫 번째 입력은 sum_trampoline(n, 0)입니다.

그래서 코드는 다음과 같습니다.

<code class="language-java">Object sum_trampoline_bootstrap(int n) {
    return sum_trampoline(n, 0);
}

Object sum_trampoline(int n, int acc) {
    if (n == 0) {
        return acc;
    }
    return (Supplier<object>) () -> sum(n - 1, acc + n);
}</code>
로그인 후 복사
로그인 후 복사

유형을 사용하여 스프링보드 설명

스프링보드는 대략 다음과 같은 형식이어야 합니다.

<code class="language-java">let trampolim = primeiraChamada(input);

while (trampolim is continuation) {
    trampolim = trampolim.continue();
}
return trampolim;</code>
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사

그러나 이는 코딩의 자유를 많이 제공하며 Java 세계에서는 그다지 직관적이지 않습니다. 객체에 물어보면 연속인지 확인할 수 있습니다. "값을 찾았나요?"라고 묻는다면 어떨까요? 또 다른 점은 Java에는 합계 유형이 없기 때문에 return trampolim는 값을 반환하는 대신 실제로 trampolim 유형을 반환한다는 것입니다. trampolim.value()으로 돌아갈 수 있습니다.

마지막으로 포인트는 스프링보드의 부트스트래핑입니다. 이를 위해 함수를 사용하여 입력을 적절한 포고 반환 값으로 변환할 수 있습니다. 더 나은 사용을 위해 입력과 결과를 일반화할 수 있습니다.

<code class="language-java">public static <R> R trampoline(IN input,
                                   Function<IN, TrampolineStep<R>> trampolinebootStrap) {
  TrampolineStep<R> nextStep = trampolinebootStrap.apply(input);
  while (!nextStep.gotValue()) {
    nextStep = nextStep.runNextStep();
  }
  return nextStep.value();
}</code>
로그인 후 복사
로그인 후 복사

TrampolineStep<R>인터페이스는 어떻습니까?

세 가지 방법을 정의합니다.

  • gotValue: 값을 찾았는지 묻습니다.
  • value: 찾은 값을 반환합니다.
  • runNextStep: 값 또는 연속을 반환합니다.

기본적으로 두 가지 상태가 있습니다.

  • 찾은 값
  • 연속입니다

따라서 정적 메소드를 사용하여 초기화할 수 있습니다. 값이 발견된 경우 값을 전달해야 합니다.

<code class="language-java">int sum(int n) {
    if (n == 0) {
        return 0;
    }
    return n + sum(n - 1);
}</code>
로그인 후 복사
로그인 후 복사

연속의 경우 다음 연속 항목을 가져오는 방법을 전달해야 합니다.

<code class="language-java">let trampolim = primeiraChamada(input);

while (trampolim is continuation) {
    trampolim = trampolim.continue();
}
return trampolim;</code>
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사

sum_trampoline이를 어떻게 달성할 수 있나요?

<code class="language-java">Object sum_trampoline_bootstrap(int n) {
    return sum_trampoline(n, 0);
}

Object sum_trampoline(int n, int acc) {
    if (n == 0) {
        return acc;
    }
    return (Supplier<object>) () -> sum(n - 1, acc + n);
}</code>
로그인 후 복사
로그인 후 복사

테일콜 피보나치

피보나치의 고전적인 구현은 재귀적 정의를 따릅니다.

<code class="language-java">let trampolim = primeiraChamada(input);

while (trampolim is continuation) {
    trampolim = trampolim.continue();
}
return trampolim;</code>
로그인 후 복사
로그인 후 복사
로그인 후 복사
로그인 후 복사

피보나치 정의를 재귀적으로 확장하지 않고 앞으로 확장하는 반복 버전도 있습니다. 0과 1에서 시작하여 해당 값에 도달할 때까지:

<code class="language-java">public static <R> R trampoline(IN input,
                                   Function<IN, TrampolineStep<R>> trampolinebootStrap) {
  TrampolineStep<R> nextStep = trampolinebootStrap.apply(input);
  while (!nextStep.gotValue()) {
    nextStep = nextStep.runNextStep();
  }
  return nextStep.value();
}</code>
로그인 후 복사
로그인 후 복사

"테일 호출 재귀"를 사용하는 이 구현의 정방향 버전이 있습니다.

<code class="language-java">static <X> TrampolineStep<X> valueFound(X value) {
    return new TrampolineStep() {
        @Override
        public boolean gotValue() {
            return true;
        }

        @Override
        public X value() {
            return value;
        }

        @Override
        public TrampolineStep<X> runNextStep() {
            return this;
        }
    };
}</code>
로그인 후 복사

여기에서는 테일 콜 재귀 피보나치에 사용될 숫자를 준비하는 입력 인터페이스를 분리합니다. 앞으로 나아가면서 fib[0] => 0, fib[1] => 1 매핑으로 시작하여 인덱스 0부터 인덱스 n에 도달할 때까지 탐색합니다.

피보나치: 테일 콜에서 스프링보드까지

fib_tc의 예는 피보나치 스프링보드를 잘 보여줍니다.

<code class="language-java">static <X> TrampolineStep<X> goonStep(Supplier<TrampolineStep<X>> x) {
    return new TrampolineStep() {
        @Override
        public boolean gotValue() {
            return false;
        }

        @Override
        public X value() {
            throw new RuntimeException("dont call this");
        }

        @Override
        public TrampolineStep<X> runNextStep() {
            return x.get();
        }
    };
}</code>
로그인 후 복사

이것은 단지 뼈대일 뿐이며 컴파일하고 실행하려면 TrampolineStep 인터페이스의 완전한 구현과 trampoline 함수의 완전한 구현이 필요합니다. 또한 IN을 특정 입력 유형으로 바꿔야 합니다.

위 내용은 트램펄린, Java의 예의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
저자별 최신 기사
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿