Skip to content
Advertisement

How to find all connected subgraphs of an undirected graph

I need some help for a problem that i am struggling to solve.

Example table:

I want to group identifiers that are related with each other between two columns and assign a unique group id.

Desired Output:

Note:the column Gr.Members is not necessary, mostly is used for a clearer view.

So the definition for a group is: A row belongs to a group if it shares at least one identifier with at least one row of this group

But the group id has to be assigned to each identifier(selected by the union of the two columns) not to the row.

Any help on how to build a query to give the desired output?

Thank you.


Update: Below are some extra sample sets with their expected output.


Given table:

Expected output: all the records should belong to the same group with group ID = 1.


Given Table:

Expected output: The records should be in the same group with group ID = 1.

Advertisement

Answer

Here is a variant that doesn’t use cursor, but uses a single recursive query.

Essentially, it treats the data as edges in a graph and traverses recursively all edges of the graph, stopping when the loop is detected. Then it puts all found loops in groups and gives each group a number.

See the detailed explanations of how it works below. I recommend you to run the query CTE-by-CTE and examine each intermediate result to understand what it does.

Sample 1

Sample 2

I added one more row with z value to have multiple rows with unpaired values.

Sample 3

Query

Result 1

Result 2

Result 3

How it works

I’ll use the second set of sample data for this explanation.

CTE_Idents

CTE_Idents gives the list of all Identifiers that appear in both Ident1 and Ident2 columns. Since they can appear in any order we UNION both columns together. UNION also removes any duplicates.

CTE_Pairs

CTE_Pairs gives the list of all edges of the graph in both directions. Again, UNION is used to remove any duplicates.

CTE_Recursive

CTE_Recursive is the main part of the query that recursively traverses the graph starting from each unique Identifier. These starting rows are produced by the first part of UNION ALL. The second part of UNION ALL recursively joins to itself linking Ident2 to Ident1. Since we pre-made CTE_Pairs with all edges written in both directions, we can always link only Ident2 to Ident1 and we’ll get all paths in the graph. At the same time the query builds IdentPath – a string of comma-delimited Identifiers that have been traversed so far. It is used in the WHERE filter:

As soon as we come across the Identifier that had been included in the Path before, the recursion stops as the list of connected nodes is exhausted. AnchorIdent is the starting Identifier for the recursion, it will be used later to group results. Lvl is not really used, I included it for better understanding of what is going on.

CTE_CleanResult

CTE_CleanResult leaves only relevant parts from CTE_Recursive and again merges both Ident1 and Ident2 using UNION.

Final SELECT

Now we need to build a string of comma-separated Ident values for each AnchorIdent. CROSS APPLY with FOR XML does it. DENSE_RANK() calculates the GroupID numbers for each AnchorIdent.

User contributions licensed under: CC BY-SA
4 People found this is helpful
Advertisement