edu.mines.jtk.interp.DiscreteSibsonGridder2 Maven / Gradle / Ivy
Show all versions of edu-mines-jtk Show documentation
/****************************************************************************
Copyright 2009, Colorado School of Mines and others.
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 edu.mines.jtk.interp;
import edu.mines.jtk.dsp.Sampling;
import static edu.mines.jtk.util.ArrayMath.*;
/**
* A discrete approximation of Sibson's natural neighbor interpolation.
* Given a set of known samples of a function f(x1,x2) for scattered points
* x1 and x2, this approximation computes an interpolating function g(x1,x2)
* with specified uniform samplings of x1 and x2. The function g(x1,x2)
* approximates the natural neighbor interpolant proposed by Sibson (1981).
*
* In this approximation, all scattered points x1 and x2 are rounded to the
* nearest uniformly sampled points. If two or more known samples fall into
* a single uniform sample bin, their values f(x1,x2) are averaged to obtain
* the value g(x1,x2) for that bin. Values of g(x1,x2) for all other bins are
* computed to approximate natural neighbor interpolation.
*
* The primary goal of this implementation is simplicity. Like the method
* of Park et al. (2006), it requires no Delaunay triangulation or Voronoi
* tesselation, and its cost decreases as the number of known samples
* increases. Moreover, unlike the method of Park et al., this method uses
* no auxilary data structure such as a k-d tree to find nearest known
* samples. Computational complexity of this method is within a constant
* factor of that of Park et al.
*
* Discrete implementations of Sibson's interpolation can produce artifacts
* (small axis-aligned ridges or valleys) caused by sampling circles on a
* rectangular grid. To attenuate these artifacts, this method applies some
* number of Gauss-Seidel iterations of bi-Laplacian smoothing to the
* interpolated samples, without modifying the known samples.
*
* References:
* Park, S.W., L. Linsen, O. Kreylos, J.D. Owens, B. Hamann, 2006,
* Discrete Sibson interpolation: IEEE Transactions on Visualization
* and Computer Graphics,
* v. 12, 243-253.
* Sibson, R., 1981, A brief description of natural neighbor interpolation,
* in V. Barnett, ed., Interpreting Multivariate Data: John Wiley and Sons,
* 21-36.
*
* @author Dave Hale, Colorado School of Mines
* @version 2009.10.10
*/
public class DiscreteSibsonGridder2 implements Gridder2 {
/**
* Constructs a nearest neighbor gridder with specified known samples.
* The specified arrays are copied; not referenced.
* @param f array of known sample values f(x1,x2).
* @param x1 array of known sample x1 coordinates.
* @param x2 array of known sample x2 coordinates.
*/
public DiscreteSibsonGridder2(float[] f, float[] x1, float[] x2) {
setScattered(f,x1,x2);
}
/**
* Sets the number of bi-Laplacian smoothing iterations.
* If non-zero, these iterations are performed at the end of discrete
* Sibson interpolation, and does not alter known sample values. This
* smoothing attenuates high-frequency artifacts caused by approximating
* circles on a uniform sampling grid. However, smoothing iterations may
* also create unwanted oscillations in gridded values. The default
* number of smoothing iterations is zero.
* @param nsmooth number of smoothing iterations.
*/
public void setSmooth(int nsmooth) {
_nsmooth = nsmooth;
}
///////////////////////////////////////////////////////////////////////////
// interface Gridder2
public void setScattered(float[] f, float[] x1, float[] x2) {
_n = f.length;
_f = f;
_x1 = x1;
_x2 = x2;
}
public float[][] grid(Sampling s1, Sampling s2) {
int n1 = s1.getCount();
int n2 = s2.getCount();
float fx1 = (float)s1.getFirst();
float fx2 = (float)s2.getFirst();
float lx1 = (float)s1.getLast();
float lx2 = (float)s2.getLast();
float dx1 = (float)s1.getDelta();
float dx2 = (float)s2.getDelta();
float od1 = 1.0f/dx1;
float od2 = 1.0f/dx2;
// Sample offsets, sorted by increasing distance. These offsets are
// used in expanding-circle searches for nearest known samples.
// We tabulate offsets for only one quadrant of a circle, because
// offsets for the other three quadrants can be found by symmetry.
int nk = n1*n2;
int[] kk = new int[nk];
float[] ds = new float[nk];
for (int i2=0,k=0; i2lx1) continue; // skip scattered values
if (x2ilx2) continue; // that fall out of bounds
int i1 = (int)(0.5f+(x1i-fx1)*od1);
int i2 = (int)(0.5f+(x2i-fx2)*od2);
c[i2][i1] += 1.0f; // count known values accumulated
g[i2][i1] += _f[i]; // accumulate known values
}
// Average where more than one known sample per bin.
for (int i2=0; i20.0f) {
g[i2][i1] /= c[i2][i1]; // normalize sum of known sample values
c[i2][i1] = -c[i2][i1]; // negative counts denote known samples
}
}
}
// For all uniform sample bins (centers of scattering circles), ...
for (int i2=0; i2=n2) continue;
for (int m1=0,j1=i1-k1k; m1<2; ++m1,j1=i1+k1k) {
if (j1<0 || j1>=n1) continue;
if (c[j2][j1]<0.0f) { // if sample is known, ...
kn = k;
fn = g[j2][j1];
}
}
}
}
// By symmetry, each pair of offset indices (k1,k2) actually
// represents four uniformly sampled points. In a rectangular
// sampling grid, either four or eight such points lie equidistant
// from the origin. If eight offsets, then four of them may
// immediately follow the four that we found in the search
// above. Here we include these next four, if necessary.
if (kn=n2) continue;
for (int m1=0,j1=i1-k1k; m1<2; ++m1,j1=i1+k1k) {
if (j1<0 || j1>=n1) continue;
if (c[j2][j1]>=0.0f) { // if sample is unknown, ...
g[j2][j1] += fn;
c[j2][j1] += 1.0f;
}
}
}
}
}
}
// Normalize accumulated values by the number scattered into each bin.
for (int i2=0; i20.0f)
g[i2][i1] /= c[i2][i1];
}
}
// Optional Gauss-Seidel iterations of bi-Laplacian smoothing to
// attenuate artifacts in discrete Sibson interpolation.
int n1m = n1-1;
int n2m = n2-1;
float a1 = 8.0f/20.0f;
float a2 = -2.0f/20.0f;
float a3 = -1.0f/20.0f;
for (int jsmooth=0; jsmooth<_nsmooth; ++jsmooth) {
for (int i2=0; i20.0f) {
float g1 = a1*(g[i2 ][i1m]+g[i2 ][i1p]+g[i2m][i1 ]+g[i2p][i1 ]);
float g2 = a2*(g[i2m][i1m]+g[i2m][i1p]+g[i2p][i1m]+g[i2p][i1p]);
float g3 = a3*(g[i2][i1mm]+g[i2][i1pp]+g[i2mm][i1]+g[i2pp][i1]);
g[i2][i1] = g1+g2+g3;
}
}
}
}
return g;
}
///////////////////////////////////////////////////////////////////////////
// private
private int _n;
private float[] _f,_x1,_x2;
private int _nsmooth;
}