All Downloads are FREE. Search and download functionalities are using the official Maven repository.

boofcv.alg.segmentation.ms.SegmentMeanShift 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.InputSanityCheck;
import boofcv.struct.ConnectRule;
import boofcv.struct.image.GrayS32;
import boofcv.struct.image.ImageBase;
import boofcv.struct.image.ImageType;
import georegression.struct.point.Point2D_I32;
import org.ddogleg.struct.FastQueue;
import org.ddogleg.struct.GrowQueue_I32;
import org.ddogleg.struct.Stoppable;

/**
 * 

* Performs mean-shift segmentation on an image. Primary based upon the description provided in [1], it first * uses mean-shift to find the mode of each pixel in the image. The mode is the location mean-shift converges to * when initialized at a particular pixel. All the pixels which have the same mode, to within tolerance, are combined * into one region. Since mean-shift does not guarantee that pixels which have the same label are connected * to each other the labeled image is now segmented to ensure connectivity between labels. If a minimum size is * specified for a segment then small regions are pruned and their pixels combined into the adjacent region which has * the most similar color. *

* *

* Mean-shift segmentation can be slow but it has the advantage of there being relatively few tuning parameters. * Few tuning parameters make it easier to work with and to some extent more robust. *

* *

* NOTE: Connectivity rule of 4 tends to produce more tightly compact regions while 8 produces fewer regions but * with a more complex surface. *

* *

* [1] Comaniciu, Dorin, and Peter Meer. "Mean shift analysis and applications." Computer Vision, 1999. * The Proceedings of the Seventh IEEE International Conference on. Vol. 2. IEEE, 1999. *

* @author Peter Abeles */ public class SegmentMeanShift> implements Stoppable { // finds mean shift modes SegmentMeanShiftSearch search; // Combines similar regions together MergeRegionMeanShift merge; // ensures that all pixels in segment are connected ClusterLabeledImage segment; // Prunes and merges smaller regions together MergeSmallRegions prune; // If a request to stop running hsa been requested volatile boolean stopRequested=false; // contains resegmented image after enforcing all points be connected // GrayS32 pixelToRegion2 = new GrayS32(1,1); /** * Specifies internal classes used by mean-shift. * * @param search mean-shift search * @param merge Used to merge regions * @param prune Prunes small regions and merges them If null then prune step will be skipped. * @param connectRule Specify if a 4 or 8 connect rule should be used when segmenting disconnected regions. Try 4 */ public SegmentMeanShift(SegmentMeanShiftSearch search, MergeRegionMeanShift merge, MergeSmallRegions prune, ConnectRule connectRule ) { this.search = search; this.merge = merge; this.prune = prune; this.segment = new ClusterLabeledImage(connectRule); } /** * Performs mean-shift segmentation on the input image. The * total number of regions can be found by calling {@link #getNumberOfRegions()}. * * @param image Image * @param output Storage for output image. Each pixel is set to the region it belongs to. */ public void process( T image , GrayS32 output ) { InputSanityCheck.checkSameShape(image,output); stopRequested = false; // long time0 = System.currentTimeMillis(); search.process(image); if( stopRequested ) return; // long time1 = System.currentTimeMillis(); FastQueue regionColor = search.getModeColor(); GrayS32 pixelToRegion = search.getPixelToRegion(); GrowQueue_I32 regionPixelCount = search.getRegionMemberCount(); FastQueue modeLocation = search.getModeLocation(); merge.process(pixelToRegion,regionPixelCount,regionColor,modeLocation); if( stopRequested ) return; // long time2 = System.currentTimeMillis(); segment.process(pixelToRegion, output, regionPixelCount); if( stopRequested ) return; // long time3 = System.currentTimeMillis(); if( prune != null) prune.process(image,output,regionPixelCount,regionColor); // long time4 = System.currentTimeMillis(); // System.out.println("Search: "+(time1-time0)+" Merge: "+(time2-time1)+ // " segment: "+(time3-time2)+" Prune: "+(time4-time3)); } /** * The number of regions which it found in the image. * @return Total regions */ public int getNumberOfRegions() { return search.getRegionMemberCount().size; } /** * Average color of each region */ public FastQueue getRegionColor() { return search.getModeColor(); } /** * Number of pixels in each region */ public GrowQueue_I32 getRegionSize() { return search.getRegionMemberCount(); } public ImageType getImageType() { return search.getImageType(); } @Override public void requestStop() { stopRequested = true; search.requestStop(); merge.requestStop(); } @Override public boolean isStopRequested() { return stopRequested; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy