Мне кажется, что должно быть возможно напечатать круговой связанный список назад в постоянном пространстве и линейном времени с помощью рекурсии и оптимизации хвоста. Тем не менее, у меня возникают трудности из-за попытки распечатать текущий элемент после выполнения рекурсивного вызова. Проверяя разборку, я вижу, что функция вызывается и не подпрыгивает. Если я изменил его для печати вперед, а не назад, вызов функции будет правильно удален.Распечатайте связанный список назад в постоянном пространстве и линейном времени с помощью рекурсии
Я видел this Связанный вопрос, однако меня особенно интересует его решение с использованием рекурсии и TCO.
код я использую:
#include <stdio.h>
struct node {
int data;
struct node *next;
};
void bar(struct node *elem, struct node *sentinel)
{
if (elem->next == sentinel) {
printf("%d\n", elem->data);
return;
}
bar(elem->next, sentinel), printf("%d\n", elem->data);
}
int main(void)
{
struct node e1, e2;
e1.data = 1;
e2.data = 2;
e1.next = &e2;
e2.next = &e1;
bar(&e1, &e1);
return 0;
}
и компиляции с
$ g++ -g -O3 -Wa,-alh test.cpp -o test.o
обновление: решается с помощью ответа Джони с небольшими изменениями для кругового списка
void bar(struct node *curr, struct node *prev, struct node *sentinel,
int pass)
{
if (pass == 1) printf("%d\n", curr->data);
if (pass > 1) return;
if ((pass == 1) && (curr == sentinel))
return;
/* reverse current node */
struct node *next = curr->next;
curr->next = prev;
if (next != sentinel) {
/* tail call with current pass */
bar(next, curr, sentinel, pass);
} else if ((pass == 1) && (next == sentinel)) {
/* make sure to print the last element */
bar(next, curr, sentinel, pass);
} else {
/* end of list reached, go over list in reverse */
bar(curr, prev, sentinel, pass+1);
}
}
Проблема заключается в «хвост» часть «хвоста вызова». Оптимизация возможна только в том случае, если вызов - это последнее, что делает функция. –
В чем разница между вызовом функции и функцией перехода? Понятия не имею. – jimifiki
Вы не можете использовать tail calls для этого, потому что для печати связанного списка в обратном направлении вам нужно выполнить всю работу на обратном пути. – Medinoc