Source code for networkx.algorithms.isomorphism.tree_isomorphism

"""
An algorithm for finding if two undirected trees are isomorphic,
and if so returns an isomorphism between the two sets of nodes.

This algorithm uses a routine to tell if two rooted trees (trees with a
specified root node) are isomorphic, which may be independently useful.

This implements an algorithm from:
The Design and Analysis of Computer Algorithms
by Aho, Hopcroft, and Ullman
Addison-Wesley Publishing 1974
Example 3.2 pp. 84-86.

A more understandable version of this algorithm is described in:
Homework Assignment 5
McGill University SOCS 308-250B, Winter 2002
by Matthew Suderman
http://crypto.cs.mcgill.ca/~crepeau/CS250/2004/HW5+.pdf
"""

import networkx as nx
from networkx.utils.decorators import not_implemented_for

__all__ = ["rooted_tree_isomorphism", "tree_isomorphism"]


@nx._dispatchable(graphs={"t1": 0, "t2": 2}, returns_graph=True)
def root_trees(t1, root1, t2, root2):
    """Create a single digraph dT of free trees t1 and t2
    #   with roots root1 and root2 respectively
    # rename the nodes with consecutive integers
    # so that all nodes get a unique name between both trees

    # our new "fake" root node is 0
    # t1 is numbers from 1 ... n
    # t2 is numbered from n+1 to 2n
    """

    dT = nx.DiGraph()

    newroot1 = 1  # left root will be 1
    newroot2 = nx.number_of_nodes(t1) + 1  # right will be n+1

    # may be overlap in node names here so need separate maps
    # given the old name, what is the new
    namemap1 = {root1: newroot1}
    namemap2 = {root2: newroot2}

    # add an edge from our new root to root1 and root2
    dT.add_edge(0, namemap1[root1])
    dT.add_edge(0, namemap2[root2])

    for i, (v1, v2) in enumerate(nx.bfs_edges(t1, root1)):
        namemap1[v2] = i + namemap1[root1] + 1
        dT.add_edge(namemap1[v1], namemap1[v2])

    for i, (v1, v2) in enumerate(nx.bfs_edges(t2, root2)):
        namemap2[v2] = i + namemap2[root2] + 1
        dT.add_edge(namemap2[v1], namemap2[v2])

    # now we really want the inverse of namemap1 and namemap2
    # giving the old name given the new
    # since the values of namemap1 and namemap2 are unique
    # there won't be collisions
    namemap = {}
    for old, new in namemap1.items():
        namemap[new] = old
    for old, new in namemap2.items():
        namemap[new] = old

    return (dT, namemap, newroot1, newroot2)


# figure out the level of each node, with 0 at root
@nx._dispatchable
def assign_levels(G, root):
    level = {}
    level[root] = 0
    for v1, v2 in nx.bfs_edges(G, root):
        level[v2] = level[v1] + 1

    return level


# now group the nodes at each level
def group_by_levels(levels):
    L = {}
    for n, lev in levels.items():
        if lev not in L:
            L[lev] = []
        L[lev].append(n)

    return L


# now lets get the isomorphism by walking the ordered_children
def generate_isomorphism(v, w, M, ordered_children):
    # make sure tree1 comes first
    assert v < w
    M.append((v, w))
    for i, (x, y) in enumerate(zip(ordered_children[v], ordered_children[w])):
        generate_isomorphism(x, y, M, ordered_children)


[docs] @nx._dispatchable(graphs={"t1": 0, "t2": 2}) def rooted_tree_isomorphism(t1, root1, t2, root2): """ Given two rooted trees `t1` and `t2`, with roots `root1` and `root2` respectively this routine will determine if they are isomorphic. These trees may be either directed or undirected, but if they are directed, all edges should flow from the root. It returns the isomorphism, a mapping of the nodes of `t1` onto the nodes of `t2`, such that two trees are then identical. Note that two trees may have more than one isomorphism, and this routine just returns one valid mapping. Parameters ---------- `t1` : NetworkX graph One of the trees being compared `root1` : a node of `t1` which is the root of the tree `t2` : undirected NetworkX graph The other tree being compared `root2` : a node of `t2` which is the root of the tree This is a subroutine used to implement `tree_isomorphism`, but will be somewhat faster if you already have rooted trees. Returns ------- isomorphism : list A list of pairs in which the left element is a node in `t1` and the right element is a node in `t2`. The pairs are in arbitrary order. If the nodes in one tree is mapped to the names in the other, then trees will be identical. Note that an isomorphism will not necessarily be unique. If `t1` and `t2` are not isomorphic, then it returns the empty list. """ assert nx.is_tree(t1) assert nx.is_tree(t2) # get the rooted tree formed by combining them # with unique names (dT, namemap, newroot1, newroot2) = root_trees(t1, root1, t2, root2) # compute the distance from the root, with 0 for our levels = assign_levels(dT, 0) # height h = max(levels.values()) # collect nodes into a dict by level L = group_by_levels(levels) # each node has a label, initially set to 0 label = {v: 0 for v in dT} # and also ordered_labels and ordered_children # which will store ordered tuples ordered_labels = {v: () for v in dT} ordered_children = {v: () for v in dT} # nothing to do on last level so start on h-1 # also nothing to do for our fake level 0, so skip that for i in range(h - 1, 0, -1): # update the ordered_labels and ordered_children # for any children for v in L[i]: # nothing to do if no children if dT.out_degree(v) > 0: # get all the pairs of labels and nodes of children # and sort by labels s = sorted((label[u], u) for u in dT.successors(v)) # invert to give a list of two tuples # the sorted labels, and the corresponding children ordered_labels[v], ordered_children[v] = list(zip(*s)) # now collect and sort the sorted ordered_labels # for all nodes in L[i], carrying along the node forlabel = sorted((ordered_labels[v], v) for v in L[i]) # now assign labels to these nodes, according to the sorted order # starting from 0, where identical ordered_labels get the same label current = 0 for i, (ol, v) in enumerate(forlabel): # advance to next label if not 0, and different from previous if (i != 0) and (ol != forlabel[i - 1][0]): current += 1 label[v] = current # they are isomorphic if the labels of newroot1 and newroot2 are 0 isomorphism = [] if label[newroot1] == 0 and label[newroot2] == 0: generate_isomorphism(newroot1, newroot2, isomorphism, ordered_children) # get the mapping back in terms of the old names # return in sorted order for neatness isomorphism = [(namemap[u], namemap[v]) for (u, v) in isomorphism] return isomorphism
[docs] @not_implemented_for("directed") @not_implemented_for("multigraph") @nx._dispatchable(graphs={"t1": 0, "t2": 1}) def tree_isomorphism(t1, t2): """ Given two undirected (or free) trees `t1` and `t2`, this routine will determine if they are isomorphic. It returns the isomorphism, a mapping of the nodes of `t1` onto the nodes of `t2`, such that two trees are then identical. Note that two trees may have more than one isomorphism, and this routine just returns one valid mapping. Parameters ---------- t1 : undirected NetworkX graph One of the trees being compared t2 : undirected NetworkX graph The other tree being compared Returns ------- isomorphism : list A list of pairs in which the left element is a node in `t1` and the right element is a node in `t2`. The pairs are in arbitrary order. If the nodes in one tree is mapped to the names in the other, then trees will be identical. Note that an isomorphism will not necessarily be unique. If `t1` and `t2` are not isomorphic, then it returns the empty list. Notes ----- This runs in O(n*log(n)) time for trees with n nodes. """ assert nx.is_tree(t1) assert nx.is_tree(t2) # To be isomorphic, t1 and t2 must have the same number of nodes. if nx.number_of_nodes(t1) != nx.number_of_nodes(t2): return [] # Another shortcut is that the sorted degree sequences need to be the same. degree_sequence1 = sorted(d for (n, d) in t1.degree()) degree_sequence2 = sorted(d for (n, d) in t2.degree()) if degree_sequence1 != degree_sequence2: return [] # A tree can have either 1 or 2 centers. # If the number doesn't match then t1 and t2 are not isomorphic. center1 = nx.center(t1) center2 = nx.center(t2) if len(center1) != len(center2): return [] # If there is only 1 center in each, then use it. if len(center1) == 1: return rooted_tree_isomorphism(t1, center1[0], t2, center2[0]) # If there both have 2 centers, then try the first for t1 # with the first for t2. attempts = rooted_tree_isomorphism(t1, center1[0], t2, center2[0]) # If that worked we're done. if len(attempts) > 0: return attempts # Otherwise, try center1[0] with the center2[1], and see if that works return rooted_tree_isomorphism(t1, center1[0], t2, center2[1])