Skip to content

Commit

Permalink
ENH: Improve partial diagrams algo (#116)
Browse files Browse the repository at this point in the history
* Improves performance of
`diagrams.generate_partial_diagrams`
by efficiently filtering out invalid
edges that do not span all nodes in
the kinetic diagram. For small diagrams
the performance benefit is null, but
for more complex diagrams (e.g., the
8-state model of EmrE), there is a
statistically significant performance
increase.

* Fixes #22
  • Loading branch information
nawtrey authored Aug 21, 2024
1 parent 190f3ae commit 608b1a1
Showing 1 changed file with 20 additions and 13 deletions.
33 changes: 20 additions & 13 deletions kda/diagrams.py
Original file line number Diff line number Diff line change
Expand Up @@ -350,19 +350,26 @@ def generate_partial_diagrams(G, return_edges=False):
# get list of possible combinations of unique edges (N choose N-1)
# and iterate over each unique combination
for edge_list in itertools.combinations(unique_edges, n_edges):
# make a base partial graph
partial = nx.Graph()
partial.add_nodes_from(base_nodes)
partial.add_edges_from(edge_list)
# for the tree to be valid, it must have N-1 edges and
# be connected. Since we already have the edge list
# generated for N-1 edges, just check if it is connected
if nx.is_connected(partial):
if return_edges:
partials[i, :] = edge_list
else:
partials[i] = partial
i += 1
# partial diagrams must span all nodes in the kinetic diagram
# check if edges include all nodes in the base diagram
# before continuing
nodes_in_edges = set([item for edge in edge_list for item in edge])
nodes_span_edges = set(base_nodes) == nodes_in_edges
# filter out cases where edges don't span all nodes
if nodes_span_edges:
# make a base partial graph
partial = nx.Graph()
partial.add_nodes_from(base_nodes)
partial.add_edges_from(edge_list)
# spanning trees must have `N-1` edges that span all nodes
# and are connected. We already have `N-1` edges that
# span all nodes in `G`, just check if they are connected
if nx.is_connected(partial):
if return_edges:
partials[i, :] = edge_list
else:
partials[i] = partial
i += 1

return partials

Expand Down

0 comments on commit 608b1a1

Please sign in to comment.