Jump to content

Перекрывающиеся подзадачи

В информатике , что проблема говорят имеет перекрывающиеся подзадачи , если проблему можно разбить на подзадачи, которые используются повторно несколько раз, или рекурсивный алгоритм задачи решает одну и ту же подзадачу снова и снова, а не всегда генерирует новые подзадачи. [1] [2] [3]

Например, проблема вычисления последовательности Фибоначчи имеет перекрывающиеся подзадачи. Задачу вычисления n-го числа Фибоначчи F ( n ) можно разбить на подзадачи вычисления F ( n - 1) и F ( n - 2), а затем их сложения. Подзадачу вычисления F ( n − 1) можно разбить на подзадачу, которая включает в себя вычисление F ( n − 2). Следовательно, вычисление F ( n - 2) используется повторно, и, таким образом, последовательность Фибоначчи демонстрирует перекрывающиеся подзадачи.

Наивный рекурсивный подход к такой проблеме обычно терпит неудачу из-за экспоненциальной сложности . Если проблема также имеет оптимальные свойства подструктуры , динамическое программирование — хороший способ ее решить.

Пример последовательности Фибоначчи в C

[ редактировать ]

Рассмотрим следующий код C :

#include <stdio.h>

#define N 5

static int fibMem[N];

int fibonacci(int n) {
	int r = 1;
	if (n > 2) {
		r = fibonacci(n - 1) + fibonacci(n - 2);
	}
	fibMem[n - 1] = r;
	return r;
}

void printFibonacci() {
    int i;
    for (i = 1; i <= N; i++) {
        printf("fibonacci(%d): %d\n", i, fibMem[i - 1]);
    }
}

int main(void) {
    fibonacci(N);
	printFibonacci();
	return 0;
}

/* Output:
    fibonacci(1): 1
    fibonacci(2): 1
    fibonacci(3): 2
    fibonacci(4): 3
    fibonacci(5): 5 */

При выполнении fibonacci Функция вычисляет значение некоторых чисел в последовательности много раз, следуя шаблону, который можно визуализировать с помощью этой диаграммы:

f(5) = f(4) + f(3) = 5
       |      |
       |      f(3) = f(2) + f(1) = 2
       |             |      |
       |             |      f(1) = 1
       |             |
       |             f(2) = 1
       |
       f(4) = f(3) + f(2) = 3
              |      |
              |      f(2) = 1
              |
              f(3) = f(2) + f(1) = 2
                     |      |
                     |      f(1) = 1
                     |
                     f(2) = 1

Однако мы можем воспользоваться преимуществами мемоизации и изменить fibonacci функция, позволяющая использовать fibMem вот так:

int fibonacci(int n) {
	int r = 1;
	if (fibMem[n - 1] != 0) {
		r = fibMem[n - 1];
	} else {
		if (n > 2) {
			r = fibonacci(n - 1) + fibonacci(n - 2);
		}
		fibMem[n - 1] = r;
	}
	return r;
}

Это гораздо более эффективно, поскольку если значение r уже рассчитано на определенный n и хранится в fibMem[n - 1], функция может просто вернуть сохраненное значение, а не выполнять дополнительные рекурсивные вызовы функций. В результате получается закономерность, которую можно визуализировать с помощью этой диаграммы:

f(5) = f(4) + f(3) = 5
       |      |
       f(4) = f(3) + f(2) = 3
              |      |
              f(3) = f(2) + f(1) = 2
                     |      |
                     |      f(1) = 1
                     |
                     f(2) = 1

Разница может показаться не слишком значительной, если N 5, но по мере увеличения его значения сложность оригинала fibonacci Функция увеличивается экспоненциально, тогда как пересмотренная версия увеличивается более линейно.

См. также

[ редактировать ]
  1. ^ Введение в алгоритмы , 2-е изд. (Кормен, Лейзерсон, Ривест и Штейн) 2001, стр. 327. ISBN   0-262-03293-7 .
  2. ^ Введение в алгоритмы , 3-е изд., (Кормен, Лейзерсон, Ривест и Штейн) 2014, стр. 384. ISBN   9780262033848 .
  3. ^ Динамическое программирование: перекрывающиеся подзадачи, оптимальная подструктура , Видео MIT.
Arc.Ask3.Ru: конец переведенного документа.
Arc.Ask3.Ru
Номер скриншота №: db754df89e0d1fc7c80249b86737fae5__1680440400
URL1:https://arc.ask3.ru/arc/aa/db/e5/db754df89e0d1fc7c80249b86737fae5.html
Заголовок, (Title) документа по адресу, URL1:
Overlapping subproblems - Wikipedia
Данный printscreen веб страницы (снимок веб страницы, скриншот веб страницы), визуально-программная копия документа расположенного по адресу URL1 и сохраненная в файл, имеет: квалифицированную, усовершенствованную (подтверждены: метки времени, валидность сертификата), открепленную ЭЦП (приложена к данному файлу), что может быть использовано для подтверждения содержания и факта существования документа в этот момент времени. Права на данный скриншот принадлежат администрации Ask3.ru, использование в качестве доказательства только с письменного разрешения правообладателя скриншота. Администрация Ask3.ru не несет ответственности за информацию размещенную на данном скриншоте. Права на прочие зарегистрированные элементы любого права, изображенные на снимках принадлежат их владельцам. Качество перевода предоставляется как есть. Любые претензии, иски не могут быть предъявлены. Если вы не согласны с любым пунктом перечисленным выше, вы не можете использовать данный сайт и информация размещенную на нем (сайте/странице), немедленно покиньте данный сайт. В случае нарушения любого пункта перечисленного выше, штраф 55! (Пятьдесят пять факториал, Денежную единицу (имеющую самостоятельную стоимость) можете выбрать самостоятельно, выплаичвается товарами в течение 7 дней с момента нарушения.)