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

boofcv.alg.feature.describe.DescribePointSurfMod Maven / Gradle / Ivy

/*
 * Copyright (c) 2011-2017, 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.feature.describe;

import boofcv.factory.filter.kernel.FactoryKernelGaussian;
import boofcv.struct.convolve.Kernel2D_F64;
import boofcv.struct.image.ImageGray;
import boofcv.struct.sparse.GradientValue;
import boofcv.struct.sparse.SparseImageGradient;

/**
 * 

* Modified SURF descriptor which attempts to smooth out edge conditions. Based upon MU-SURF described in * [1] it computes features in over lapping sub-regions and has a separate set of weights for the large grid and * for sub-regions. Due to these improvements it will in general produce better results than {@link DescribePointSurf} * at the cost of additional computations. *

*

* [1] M. Agrawal, K. Konolige, and M. Blas, "CenSurE: Center Surround Extremas for Realtime Feature Detection and * Matching," Computer Vision – ECCV 2008 *

* * @author Peter Abeles */ public class DescribePointSurfMod> extends DescribePointSurf { // how many sample points sub-regions overlap. private int overLap; // used to weigh feature computation private Kernel2D_F64 weightGrid; private Kernel2D_F64 weightSub; private double samplesX[]; private double samplesY[]; /** * Creates a SURF descriptor of arbitrary dimension by changing how the local region is sampled. * * @param widthLargeGrid Number of sub-regions wide the large grid is. Typically 4. * @param widthSubRegion Number of sample points wide a sub-region is. Typically 5. * @param widthSample The size of a sample point. Typically 3. * @param overLap Number of sample points sub-regions overlap, Typically 2. * @param sigmaLargeGrid Sigma used to weight points in the large grid. Typically 2.5 * @param sigmaSubRegion Sigma used to weight points in the sub-region grid. Typically 2.5 * @param useHaar If true the Haar wavelet will be used (what was used in [1]), false means an image gradient * approximation will be used. True is recommended. */ public DescribePointSurfMod(int widthLargeGrid, int widthSubRegion, int widthSample, int overLap , double sigmaLargeGrid , double sigmaSubRegion , boolean useHaar, Class imageType ) { super(widthLargeGrid, widthSubRegion, widthSample, 1, useHaar,imageType); this.overLap = overLap; weightGrid = FactoryKernelGaussian.gaussianWidth(sigmaLargeGrid, widthLargeGrid); weightSub = FactoryKernelGaussian.gaussianWidth(sigmaSubRegion, widthSubRegion + 2 * overLap); double div = weightGrid.get(weightGrid.getRadius(),weightGrid.getRadius()); for( int i = 0; i < weightGrid.data.length; i++ ) weightGrid.data[i] /= div; div = weightSub.get(weightSub.getRadius(),weightSub.getRadius()); for( int i = 0; i < weightSub.data.length; i++ ) weightSub.data[i] /= div; // how wide the grid is that's being sampled in units of samples int sampleWidth = widthLargeGrid*widthSubRegion+overLap*2; samplesX = new double[sampleWidth*sampleWidth]; samplesY = new double[sampleWidth*sampleWidth]; radiusDescriptor = (widthLargeGrid*widthSubRegion)/2+overLap; } /** * Create a SURF-64 descriptor. See [1] for details. */ public DescribePointSurfMod(Class imageType) { this(4,5,3,2, 2.5 , 2.5 , false ,imageType); } /** *

* An improved SURF descriptor as presented in CenSurE paper. The sub-regions now overlap and more * points are sampled in the sub-region to allow overlap. *

* * @param c_x Center of the feature x-coordinate. * @param c_y Center of the feature y-coordinate. * @param c cosine of the orientation * @param s sine of the orientation * @param scale The scale of the wavelets. * @param features Where the features are written to. Must be 4*(widthLargeGrid*widthSubRegion)^2 large. */ public void features(double c_x, double c_y, double c , double s, double scale, SparseImageGradient gradient , double[] features) { int regionSize = widthLargeGrid*widthSubRegion; int totalSampleWidth = widthSubRegion+overLap*2; int regionR = regionSize/2; int regionEnd = regionSize-regionR; int sampleGridWidth = regionSize+2*overLap; int regionIndex = 0; // when computing the pixel coordinates it is more precise to round to the nearest integer // since pixels are always positive round() is equivalent to adding 0.5 and then converting // to an int, which floors the variable. c_x += 0.5; c_y += 0.5; // first sample the whole grid at once to avoid sampling overlapping regions twice int index = 0; for( int rY = -regionR-overLap; rY < regionEnd+overLap; rY++) { double regionY = rY*scale; for( int rX = -regionR-overLap; rX < regionEnd+overLap; rX++,index++ ) { double regionX = rX*scale; // rotate the pixel along the feature's direction int pixelX = (int)(c_x + c*regionX - s*regionY); int pixelY = (int)(c_y + s*regionX + c*regionY); GradientValue g = gradient.compute(pixelX,pixelY); samplesX[index] = g.getX(); samplesY[index] = g.getY(); } } // compute descriptor using precomputed samples int indexGridWeight = 0; for( int rY = -regionR; rY < regionEnd; rY += widthSubRegion ) { for( int rX = -regionR; rX < regionEnd; rX += widthSubRegion ) { double sum_dx = 0, sum_dy=0, sum_adx=0, sum_ady=0; // compute and sum up the response inside the sub-region for( int i = 0; i < totalSampleWidth; i++ ) { index = (rY+regionR+i)*sampleGridWidth + rX+regionR; for( int j = 0; j < totalSampleWidth; j++ , index++ ) { double w = weightSub.get(j,i); double dx = w*samplesX[index]; double dy = w*samplesY[index]; // align the gradient along image patch // note the transform is transposed double pdx = c*dx + s*dy; double pdy = -s*dx + c*dy; sum_dx += pdx; sum_adx += Math.abs(pdx); sum_dy += pdy; sum_ady += Math.abs(pdy); } } double w = weightGrid.data[indexGridWeight++]; features[regionIndex++] = w*sum_dx; features[regionIndex++] = w*sum_adx; features[regionIndex++] = w*sum_dy; features[regionIndex++] = w*sum_ady; } } } @Override public int getCanonicalWidth() { return super.getCanonicalWidth() + 2*overLap; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy