edu.mines.jtk.interp.BicubicInterpolator2 Maven / Gradle / Ivy
/****************************************************************************
Copyright 2012, 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 edu.mines.jtk.util.Check;
import static edu.mines.jtk.util.ArrayMath.*;
/**
* Piecewise bicubic polynomial interpolation of a function y(x1,x2).
* The interpolated function has continuous first derivatives dy/d1, dy/d2 and
* cross-derivatives ddy/d1d2. First derivatives are computed using methods
* that may be specified for each dimension. Cross-derivatives are computed by
* averaging derivatives of first derivatives.
*
* The function y(x1,x2) is specified by samples on a regular grid, which need
* not be uniform. The regular grid is specified by one-dimensional arrays of
* monotonically increasing coordinates x1 and x2, such that gridded x1 are
* identical for all gridded x2, and gridded x2 are identical for all gridded
* x1.
*
* Extrapolation (that is, interpolation outside the specified grid of (x1,x2)
* coordinates), is performed using cubic polynomials for the nearest grid
* samples. Extrapolated values can be well outside the [min,max] range of
* interpolated values, and should typically be avoided.
*
* @author Dave Hale, Colorado School of Mines
* @version 2012.12.27
*/
public class BicubicInterpolator2 {
/**
* Method used to compute first derivatives.
*/
public enum Method {
/**
* Uses monotonicity-preserving cubic interpolation.
* This is the default method.
*/
MONOTONIC,
/**
* Uses cubic spline interpolation.
*/
SPLINE
}
/**
* Constructs an interpolator for specified values y(x1,x2).
* @param x1 array[n1] of x1 coordinates; monotonically increasing.
* @param x2 array[n2] of x2 coordinates; monotonically increasing.
* @param y array[n2][n1] of sampled values y(x1,x2).
*/
public BicubicInterpolator2(float[] x1, float[] x2, float[][] y) {
this(Method.MONOTONIC,Method.MONOTONIC,x1,x2,y);
}
/**
* Constructs an interpolator for specified methods and values.
* @param method1 method used to compute dy/d1.
* @param method2 method used to compute dy/d2.
* @param x1 array[n1] of x1 coordinates; monotonically increasing.
* @param x2 array[n2] of x2 coordinates; monotonically increasing.
* @param y array[n2][n1] of sampled values y(x1,x2).
*/
public BicubicInterpolator2(
Method method1, Method method2,
float[] x1, float[] x2, float[][] y)
{
this(method1,method2,x1.length,x2.length,x1,x2,y);
}
/**
* Constructs an interpolator for specified methods and values.
* @param method1 method used to compute dy/d1.
* @param method2 method used to compute dy/d2.
* @param n1 number of x1 coordinates.
* @param n2 number of x2 coordinates.
* @param x1 array[n1] of x1 coordinates; monotonically increasing.
* @param x2 array[n2] of x2 coordinates; monotonically increasing.
* @param y array[n2][n1] of sampled values y(x1,x2).
*/
public BicubicInterpolator2(
Method method1, Method method2,
int n1, int n2, float[] x1, float[] x2, float[][] y)
{
Check.argument(isMonotonic(x1), "array x1 is monotonic");
Check.argument(isMonotonic(x2), "array x2 is monotonic");
_n1 = n1;
_n2 = n2;
_x1 = copy(n1,x1);
_x2 = copy(n2,x2);
_a = makeCoefficients(method1,method2,n1,n2,x1,x2,y);
}
/**
* Returns interpolated value y.
* Same as {@link #interpolate00(float,float)}.
* @param x1 coordinate x1.
* @param x2 coordinate x2.
* @return interpolated y(x1,x2).
*/
public float interpolate(float x1, float x2) {
return interpolate00(x1,x2);
}
/**
* Returns interpolated value y.
* @param x1 coordinate x1.
* @param x2 coordinate x2.
* @return interpolated y(x1,x2).
*/
public float interpolate00(float x1, float x2) {
return interpolate00(x1,x2,_ks);
}
/**
* Returns interpolated partial derivative dy/d1.
* @param x1 coordinate x1.
* @param x2 coordinate x2.
* @return interpolated dy/d1.
*/
public float interpolate10(float x1, float x2) {
return interpolate10(x1,x2,_ks);
}
/**
* Returns interpolated partial derivative dy/d2.
* @param x1 coordinate x1.
* @param x2 coordinate x2.
* @return interpolated dy/d2.
*/
public float interpolate01(float x1, float x2) {
return interpolate01(x1,x2,_ks);
}
/**
* Returns an array of interpolated values y.
* Same as {@link #interpolate00(Sampling,Sampling)}.
* @param s1 sampling of coordinate x1.
* @param s2 sampling of coordinate x2.
* @return interpolated y(x1,x2).
*/
public float[][] interpolate(Sampling s1, Sampling s2) {
return interpolate00(s1,s2);
}
/**
* Returns an array of interpolated values y.
* @param s1 sampling of coordinate x1.
* @param s2 sampling of coordinate x2.
* @return interpolated y(x1,x2).
*/
public float[][] interpolate00(Sampling s1, Sampling s2) {
int n1 = s1.getCount();
int n2 = s2.getCount();
float[][] y = new float[n2][n1];
interpolate00(s1,s2,y);
return y;
}
/**
* Computes an array of interpolated values y.
* Same as {@link #interpolate00(Sampling,Sampling,float[][])}.
* @param s1 sampling of coordinate x1.
* @param s2 sampling of coordinate x2.
* @param y output array of interpolated y(x1,x2).
*/
public void interpolate(Sampling s1, Sampling s2, float[][] y) {
interpolate00(s1,s2,y);
}
/**
* Computes an array of interpolated values y.
* @param s1 sampling of coordinate x1.
* @param s2 sampling of coordinate x2.
* @param y output array of interpolated y(x1,x2).
*/
public void interpolate00(Sampling s1, Sampling s2, float[][] y) {
int n1 = s1.getCount();
int n2 = s2.getCount();
int[] k1 = makeIndices(s1,_x1);
int[] k2 = makeIndices(s2,_x2);
for (int i2=0; i2=xs.length-1)
i = xs.length-2;
return i;
}
private void updateIndices(float x1, float x2, int[] ks) {
ks[0] = index(x1,_x1,ks[0]);
ks[1] = index(x2,_x2,ks[1]);
}
private static int[] makeIndices(float[] xi, float[] xs) {
int n = xi.length;
int[] ki = new int[n];
ki[0] = index(xi[0],xs,0);
for (int i=1; i