This repository was archived by the owner on Oct 22, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 6
[WIP] Feature cardinality matching #18
Open
Wikunia
wants to merge
4
commits into
JuliaGraphs:master
Choose a base branch
from
Wikunia:feature-cardinality-matching
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from all commits
Commits
Show all changes
4 commits
Select commit
Hold shift + click to select a range
bc29035
first version of cardinality matching
Wikunia a0e4302
Merge branch 'master' into feature-cardinality-matching
Wikunia 2c61065
test cases, bugfix in backtrack path and -1 in matching
Wikunia 5cec648
function maximal matching + boolean for level
Wikunia File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,130 @@ | ||
""" | ||
maximum_cardinality_matching(g::AbstractGraph{U}) where {U<:Integer} | ||
|
||
Given a graph `g` returns a maximum cardinality matching. | ||
This is the same as running `maximum_weight_matching` with default weights (`1`) | ||
but is faster without needing a JuMP solver. | ||
|
||
Returns MatchingResult containing: | ||
- the maximum cardinality that can be achieved | ||
- a list of each vertex's match (or -1 for unmatched vertices) | ||
""" | ||
function maximum_cardinality_matching(g::AbstractGraph{U}) where {U<:Integer} | ||
n = nv(g) | ||
init_matching = maximal_matching(g) | ||
|
||
matching = init_matching.mate | ||
matching_len = init_matching.weight | ||
|
||
# the number of edges that can possibly be part of a matching | ||
max_generally_possible = fld(n, 2) | ||
|
||
# the maximal matching is a maximum matching | ||
if matching_len == max_generally_possible | ||
return init_matching | ||
end | ||
# => there are at least two free vertices | ||
|
||
parents = zeros(Int, n) | ||
visited = falses(n) | ||
|
||
cur_level = Vector{Int}() | ||
sizehint!(cur_level, n) | ||
next_level = Vector{Int}() | ||
sizehint!(next_level, n) | ||
|
||
@inbounds while matching_len < max_generally_possible | ||
# get starting free vertex | ||
free_vertex = 0 | ||
found = false | ||
for v in vertices(g) | ||
if matching[v] == -1 | ||
visited[v] = true | ||
free_vertex = v | ||
push!(cur_level, v) | ||
|
||
odd_level = true | ||
found = false | ||
# find augmenting path | ||
while !isempty(cur_level) | ||
for v in cur_level | ||
if odd_level | ||
for t in outneighbors(g, v) | ||
# found an augmenting path if connected to a free vertex | ||
if matching[t] == -1 | ||
# traverse the augmenting path backwards and change the matching | ||
current_src = t | ||
current_dst = v | ||
back_odd_level = true | ||
while current_dst != 0 | ||
# add every second edge to the matching (this also overwrites the current matching) | ||
if back_odd_level | ||
matching[current_src] = current_dst | ||
matching[current_dst] = current_src | ||
end | ||
current_src = current_dst | ||
current_dst = parents[current_dst] | ||
back_odd_level = !back_odd_level | ||
end | ||
# added exactly one edge to the matching | ||
matching_len += 1 | ||
# terminate current search | ||
found = true | ||
break | ||
end | ||
|
||
# use a non matching edge | ||
if !visited[t] && matching[v] != t | ||
visited[t] = true | ||
parents[t] = v | ||
push!(next_level, t) | ||
end | ||
end | ||
found && break | ||
else # use a matching edge | ||
t = matching[v] | ||
if !visited[t] | ||
visited[t] = true | ||
parents[t] = v | ||
push!(next_level, t) | ||
end | ||
end | ||
end | ||
|
||
empty!(cur_level) | ||
cur_level, next_level = next_level, cur_level | ||
|
||
odd_level = !odd_level | ||
found && break | ||
end # end finding augmenting path | ||
found && break | ||
parents .= 0 | ||
visited .= false | ||
end | ||
end | ||
# checked all free vertices: | ||
# no augmenting path found => no better matching | ||
!found && break | ||
end | ||
|
||
return MatchingResult(matching_len, matching) | ||
end | ||
|
||
function maximal_matching(g::AbstractGraph{U}) where {U<:Integer} | ||
n = nv(g) | ||
matching = fill(-1, n) | ||
|
||
# get initial matching | ||
matching_len = 0 | ||
for e in edges(g) | ||
# no self loops in matching | ||
e.src == e.dst && continue | ||
# if unmatched | ||
if matching[e.src] == -1 && matching[e.dst] == -1 | ||
matching[e.src] = e.dst | ||
matching[e.dst] = e.src | ||
matching_len += 1 | ||
end | ||
end | ||
return MatchingResult(matching_len, matching) | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe it would be a good idea, to have this as a separate function? Then it could also be used for finding a greed maximal cardinality matching and be used by other functions. We could pass it in as an optional argument or a keyword argument.
Furthermore, I was thinking, maybe it would be more efficient to loop over the vertices and then over all neighbors of that vertex? I.e something like
This approach only works, if we assume that
vertices(g)
is ordered in increasing order, but it can also be adapted, such that the increasing order is not required.Also, be careful with self-loops, these should never be in a matching.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If neighbors is implemented in a fast manner yes. Thought that looping over the edges might be faster than accessing the neighbors but yours has an earlier break. Maybe I can test it on a larger graph. Good point with self loops!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think for
SimpleGraph
the edges iterator is implemented usingneighbors
anyway.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is now a separate function but still uses
edges
for now.