Enginius/Matlab

Backward recursion

해리s 2021. 1. 29. 21:14

Tree 구조가 있을 때 각 노드가 자신의 sub-tree 노드들의 정보를 다 취합해서 갖고 있게 하고 싶을 경우가 있다. 각 노드마다 자신의 child 노드의 수가 몇 개인지 원할 때도 있고 (모든 노드가 +1의 값을 갖게 하면), 혹은 inverse dynamices를 풀 때 사용되는 recursive Newton Euler (RNE)의 inward operation을 할 때가 그렇다. 

%
%                   1
%                  /|\
%                 / | \
%                /  |  \
%               2   3   4
%              / \      |
%             5   6     7
%                      / \
%                     8   9
%
%                 1
%                /
%               2
%              / \
%             5   3
%              \   \
%               6   4
%                  /
%                 7
%                /
%               8
%                \
%                 9
%
ccc

트리 구조는 위와 같이 multiary tree 구조와 left child right sibling tree가 있다고 가정하자. 

% Initialize values
f_vals = rand(1,9);

% Left child right sibling
node_lcrs(1) = struct('child',2,'sister',0,'f',f_vals(1));
node_lcrs(2) = struct('child',5,'sister',3,'f',f_vals(2));
node_lcrs(3) = struct('child',0,'sister',4,'f',f_vals(3));
node_lcrs(4) = struct('child',7,'sister',0,'f',f_vals(4));
node_lcrs(5) = struct('child',0,'sister',6,'f',f_vals(5));
node_lcrs(6) = struct('child',0,'sister',0,'f',f_vals(6));
node_lcrs(7) = struct('child',8,'sister',0,'f',f_vals(7));
node_lcrs(8) = struct('child',0,'sister',9,'f',f_vals(8));
node_lcrs(9) = struct('child',0,'sister',0,'f',f_vals(9));
[~,node_lcrs] = foo_lcrs(node_lcrs,1);

% multiary tree
node_mltr(1) = struct('childs',[2,3,4],  'f',f_vals(1));
node_mltr(2) = struct('childs',[5,6],    'f',f_vals(2));
node_mltr(3) = struct('childs',[],       'f',f_vals(3));
node_mltr(4) = struct('childs',[7],      'f',f_vals(4));
node_mltr(5) = struct('childs',[],       'f',f_vals(5));
node_mltr(6) = struct('childs',[],       'f',f_vals(6));
node_mltr(7) = struct('childs',[8,9],    'f',f_vals(7));
node_mltr(8) = struct('childs',[],       'f',f_vals(8));
node_mltr(9) = struct('childs',[],       'f',f_vals(9));
[f,node_mltr] = foo_mltr(node_mltr,1);

for i_idx = 1:length(node_lcrs)
    fprintf('[%d] lcrs:[%.2f] / mltr:[%.2f]. \n',...
        i_idx,node_lcrs(i_idx).f_sum,node_mltr(i_idx).f_sum);
end

그럼 위와 같이 돌렸을 때, 같은 결과가 나오게 된다. 

[1] lcrs:[5.57] / mltr:[5.57]. 
[2] lcrs:[1.89] / mltr:[1.89]. 
[3] lcrs:[0.42] / mltr:[0.42]. 
[4] lcrs:[2.46] / mltr:[2.46]. 
[5] lcrs:[0.79] / mltr:[0.79]. 
[6] lcrs:[0.96] / mltr:[0.96]. 
[7] lcrs:[1.54] / mltr:[1.54]. 
[8] lcrs:[0.04] / mltr:[0.04]. 
[9] lcrs:[0.85] / mltr:[0.85]. 

먼저 left-child right-sibling tree에서 동작하는 코드는

function [f_out,node_lcrs] = foo_lcrs(node_lcrs,j)
%
% Check how the recursion works 
%
if j == 0
    f_out = 0;
    return
end
f0 = node_lcrs(j).f;
% fprintf('[foo_lcrs] A j:[%d].\n',j);

%% To child
[f1,node_lcrs] = foo_lcrs(node_lcrs,node_lcrs(j).child);
f = f0 + f1;
node_lcrs(j).f_sum = f; % append f
% fprintf('[foo_lcrs] B j:[%d].\n',j);

%% To sister
[f2,node_lcrs] = foo_lcrs(node_lcrs,node_lcrs(j).sister);
% fprintf('[foo_lcrs] C j:[%d].\n',j);
f_out = f + f2;

위와 같고, 좀 더 일반적인 multiary tree에서는 아래와 같은 코드가 같은 동작을 한다. 

function [f,node_mltr] = foo_mltr(node_mltr,j)
%
% Check how the recursion works 
%
if isempty(j)
    f = 0;
    return;
end
f0 = node_mltr(j).f;

%% To child
f = f0;
for child = node_mltr(j).childs % for each child 
    [f1,node_mltr] = foo_mltr(node_mltr,child);
    f = f + f1;
end
node_mltr(j).f_sum = f;