[Bioperl-l] Unwise elimination of nodes in B:T:Node::remove_Descendent?

Hilmar Lapp hlapp at gmx.net
Fri Feb 6 09:21:31 EST 2009


On Feb 6, 2009, at 12:49 AM, Mark A. Jensen wrote:

> A   B   C   D   E
> |5  |5  |4  |4  |
> \   /   \   /   /
>  x       y    /
>  |2      |1  /
>  \       /  /
>   \    _/  / 10
>    \  /   /
>     z    /
>     |3  /
>      \ /
>      ROOT
>
>> [...]
>> __DATA__
>> (((A:5,B:5)x:2,(C:4,D:4)y:1)z:3,E:10);
>> [...]
>
> The first problem was that the "complementary approaches" didn't give
> the same answer, and one threw an error. This is the bug in the
> code above. If you look at the tree, the nodes he really wants to  
> keep are
> the leaves [A,B,E], *plus* the internal nodes [x,y,z]; that is...
>
> $tree->splice(-keep_id=>[A,B,E,x,'y',z])
> $tree->total_branch_length
>
> doesn't throw and returns 25, which is correct.

If you replace 'y' with the root in the above, yes. Otherwise no.

> The second problem is that removing [C, D] also gives him 25, which is
> what he wanted, but is not correct.

Correct.

> [...]The problem arises in this code at the end of remove_Descendent:
>
>  # remove unecessary nodes if we have removed the part |||Node.pm  
> LINE 272
>  # which branches.
>    my $a1 = $self->ancestor;
>    if( $a1 ) {
>        my $bl = $self->branch_length || 0;
>        my @d = $self->each_Descendent;
>        if (scalar @d == 1) {
>     $d[0]->branch_length($bl + ($d[0]->branch_length || 0));
>     $a1->add_Descendent($d[0]);
>     $a1->remove_Descendent($self);
>        }
>    }
>
> When node C is removed from the example tree, the node 'y' is removed
> by this code apparently as a convenience.

I guess I'm confused by the above code snippet. If @d are the  
descendants of $a1, then I don't understand what the purpose of adding  
$d[0] as a descendant of $a1 is (after altering its branch length).  
Furthermore, if $a1 is the ancestor of $self, and $a1 has only one  
descendant, wouldn't that mean that $d[0] == $self?

What am I missing?

I also think it's a bad idea to simplify the tree (or collapse  
internal nodes of degree 1) as an implicit operation. It should be  
explicit (and I believe there is a method simplify() or something  
similar, isn't there? Ah - I see you quote contract_linear_paths()).

Furthermore, I think it's also a bad idea to remove descendant leaf  
nodes if you remove an internal node. What if you really just wanted  
to remove the internal node because, for example, its branching point  
isn't well supported? So removing node 'y' should make z a node of  
degree 3, but not remove C and D unless you ask to remove the subtree  
beginning at 'y'.

	-hilmar
-- 
===========================================================
: Hilmar Lapp  -:-  Durham, NC  -:-  hlapp at gmx dot net :
===========================================================






More information about the Bioperl-l mailing list