boofcv.alg.segmentation.ms.RegionMergeTree Maven / Gradle / Ivy
/*
* Copyright (c) 2011-2018, Peter Abeles. All Rights Reserved.
*
* This file is part of BoofCV (http://boofcv.org).
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package boofcv.alg.segmentation.ms;
import boofcv.alg.filter.binary.BinaryImageOps;
import boofcv.struct.image.GrayS32;
import org.ddogleg.struct.GrowQueue_I32;
import org.ddogleg.struct.Stoppable;
/**
* Merges regions together quickly and efficiently using a directed tree graph. To merge two segments together
* first call {@link #markMerge}. Then after all the regions which are to be merged are marked call
* {@link #performMerge}.
*
* Internally a disjoint-set forest tree graph is maintained using an array. When two regions are marked to be merged
* (set-union) path-compression is done. After merging hsa finished, the graph is fully compressed so that all nodes
* point to their root directly. Then the output is computed.
*
* @author Peter Abeles
*/
public class RegionMergeTree implements Stoppable {
// list used to convert the original region ID's into their new compacted ones
// The values indicate which region a region is to be merged into
// An value of equal to its index indicates that the region is a root in the graph and
// is not to be merged with any others
protected GrowQueue_I32 mergeList = new GrowQueue_I32();
// Local copy of these lists after elements which have been merged are removed
protected GrowQueue_I32 tmpMemberCount = new GrowQueue_I32();
// the new ID of the root nodes (segments)
protected GrowQueue_I32 rootID = new GrowQueue_I32();
protected boolean stopRequested=false;
/**
* Must call before any other functions.
* @param numRegions Total number of regions.
*/
public void initializeMerge(int numRegions) {
mergeList.resize(numRegions);
for( int i = 0; i < numRegions; i++ )
mergeList.data[i] = i;
}
/**
* Merges regions together and updates the provided data structures for said changes.
*
* @param pixelToRegion (Input/Output) Image used to convert pixel location in region ID. Modified.
* @param regionMemberCount (Input/Output) List containing how many pixels belong to each region. Modified.
*/
public void performMerge( GrayS32 pixelToRegion ,
GrowQueue_I32 regionMemberCount ) {
// update member counts
flowIntoRootNode(regionMemberCount);
// re-assign the number of the root node and trim excessive nodes from the lists
setToRootNodeNewID(regionMemberCount);
// change the labels in the pixelToRegion image
BinaryImageOps.relabel(pixelToRegion, mergeList.data);
}
/**
* For each region in the merge list which is not a root node, find its root node and add to the root node
* its member count and set the index in mergeList to the root node. If a node is a root node just note
* what its new ID will be after all the other segments are removed.
*/
protected void flowIntoRootNode(GrowQueue_I32 regionMemberCount) {
rootID.resize(regionMemberCount.size);
int count = 0;
for( int i = 0; i < mergeList.size; i++ ) {
int p = mergeList.data[i];
// see if it is a root note
if( p == i ) {
// mark the root nodes new ID
rootID.data[i] = count++;
continue;
}
// traverse down until it finds the root note
int gp = mergeList.data[p];
while( gp != p ) {
p = gp;
gp = mergeList.data[p];
}
// update the count and change this node into the root node
regionMemberCount.data[p] += regionMemberCount.data[i];
mergeList.data[i] = p;
}
}
/**
* Does much of the work needed to remove the redundant segments that are being merged into their root node.
* The list of member count is updated. mergeList is updated with the new segment IDs.
*/
protected void setToRootNodeNewID( GrowQueue_I32 regionMemberCount ) {
tmpMemberCount.reset();
for( int i = 0; i < mergeList.size; i++ ) {
int p = mergeList.data[i];
if( p == i ) {
mergeList.data[i] = rootID.data[i];
tmpMemberCount.add( regionMemberCount.data[i] );
} else {
mergeList.data[i] = rootID.data[mergeList.data[i]];
}
}
regionMemberCount.reset();
regionMemberCount.addAll(tmpMemberCount);
}
/**
* This function will mark two regions for merger. Equivalent to set-union operation.
*
*
* If the two regions have yet to be merged into any others then regionB will become a member of regionA.
* Otherwise a quick heck is done to see if they are already marked for merging. If that fails it will
* traverse down the tree for each region until it gets to their roots. If the roots are not the same then
* they are merged. Either way the path is updated such that the quick check will pass.
*
*/
protected void markMerge(int regionA, int regionB) {
int dA = mergeList.data[regionA];
int dB = mergeList.data[regionB];
// Quick check to see if they reference the same node
if( dA == dB ) {
return;
}
// search down to the root node (set-find)
int rootA = regionA;
while( dA != rootA ) {
rootA = dA;
dA = mergeList.data[rootA];
}
int rootB = regionB;
while( dB != rootB ) {
rootB = dB;
dB = mergeList.data[rootB];
}
// make rootA the parent. This allows the quick test to pass in the future
mergeList.data[regionA] = rootA;
mergeList.data[regionB] = rootA;
mergeList.data[rootB] = rootA;
}
@Override
public void requestStop() {
stopRequested = true;
}
@Override
public boolean isStopRequested() {
return stopRequested;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy