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

net.finmath.marketdata.model.curves.Curve Maven / Gradle / Ivy

Go to download

finmath lib is a Mathematical Finance Library in Java. It provides algorithms and methodologies related to mathematical finance.

There is a newer version: 6.0.19
Show newest version
/*
 * (c) Copyright Christian P. Fries, Germany. Contact: [email protected].
 *
 * Created on 20.05.2005
 */
package net.finmath.marketdata.model.curves;

import java.io.IOException;
import java.io.Serializable;
import java.lang.ref.SoftReference;
import java.text.DecimalFormat;
import java.text.NumberFormat;
import java.time.LocalDate;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Locale;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

import net.finmath.interpolation.RationalFunctionInterpolation;
import net.finmath.marketdata.model.AnalyticModelInterface;
import net.finmath.time.FloatingpointDate;

/**
 * This class represents a curve build from a set of points in 2D.
 * 
 * It provides different interpolation and extrapolation methods applied to a transformation of the input point,
 * examples are
 * 
    *
  • linear interpolation of the input points
  • *
  • linear interpolation of the log of the input points
  • *
  • linear interpolation of the log of the input points divided by their respective time
  • *
  • cubic spline interpolation of the input points (or a function of the input points) (the curve will be C1).
  • *
  • Akima interpolation of the input points (or a function of the input points).
  • *
  • etc.
  • *
* *
* * For the interpolation methods provided see {@link net.finmath.marketdata.model.curves.Curve.InterpolationMethod}. * For the extrapolation methods provided see {@link net.finmath.marketdata.model.curves.Curve.ExtrapolationMethod}. * For the possible interpolation entities see {@link net.finmath.marketdata.model.curves.Curve.InterpolationEntity}. * * To construct the curve, please use the inner class CurveBuilder (a builder pattern). * * For a demo on how to construct and/or calibrate a curve see, e.g. * net.finmath.tests.marketdata.curves.CurveTest. * * @author Christian Fries */ public class Curve extends AbstractCurve implements Serializable, Cloneable { /** * Possible interpolation methods. * * @author Christian Fries */ public enum InterpolationMethod { /** Constant interpolation. Synonym of PIECEWISE_CONSTANT_LEFTPOINT. **/ PIECEWISE_CONSTANT, /** Constant interpolation. Right continuous, i.e. using the value of the left end point of the interval. **/ PIECEWISE_CONSTANT_LEFTPOINT, /** Constant interpolation using the value of the right end point of the interval. **/ PIECEWISE_CONSTANT_RIGHTPOINT, /** Linear interpolation. **/ LINEAR, /** Cubic spline interpolation. **/ CUBIC_SPLINE, /** Akima interpolation (C1 sub-spline interpolation). **/ AKIMA, /** Akima interpolation (C1 sub-spline interpolation) with a smoothing in the weights. **/ AKIMA_CONTINUOUS, /** Harmonic spline interpolation (C1 sub-spline interpolation). **/ HARMONIC_SPLINE, /** Harmonic spline interpolation (C1 sub-spline interpolation) with a monotonic filtering at the boundary points. **/ HARMONIC_SPLINE_WITH_MONOTONIC_FILTERING } /** * Possible extrapolation methods. * * @author Christian Fries */ public enum ExtrapolationMethod { /** Extrapolation using the interpolation function of the adjacent interval **/ DEFAULT, /** Constant extrapolation. **/ CONSTANT, /** Linear extrapolation. **/ LINEAR } /** * Possible interpolation entities. * @author Christian Fries */ public enum InterpolationEntity { /** Interpolation is performed on the native point values, i.e. value(t) **/ VALUE, /** Interpolation is performed on the log of the point values, i.e. log(value(t)) **/ LOG_OF_VALUE, /** Interpolation is performed on the log of the point values divided by their respective time, i.e. log(value(t))/t **/ LOG_OF_VALUE_PER_TIME } private static class Point implements Comparable, Serializable { private static final long serialVersionUID = 8857387999991917430L; public double time; public double value; public boolean isParameter; /** * @param time The time (or x-value) of the point. * @param value The value (or y-value) of the point. * @param isParameter A boolean specifying if this point is considered a "degree of freedom", e.g., in a calibration. */ public Point(double time, double value, boolean isParameter) { super(); this.time = time; this.value = value; this.isParameter = isParameter; } @Override public int compareTo(Point point) { // Ordering of the curve points with respect to time. if(this.time < point.time) return -1; if(this.time > point.time) return +1; return 0; } @Override public Object clone() { return new Point(time,value,isParameter); } } /** * A builder (following the builder pattern) for Curve objects. * Allows to successively construct a curve object by adding points. * * @author Christian Fries */ public static class CurveBuilder implements CurveBuilderInterface { private Curve curve = null; /** * Build a curve. */ public CurveBuilder() { curve = new Curve(null, null); } /** * Build a curve with a given name and given reference date. * * @param name The name of this curve. * @param referenceDate The reference date for this curve, i.e., the date which defined t=0. */ public CurveBuilder(String name, LocalDate referenceDate) { curve = new Curve(name, referenceDate); } /** * Build a curve by cloning a given curve. * * @param curve A curve to be used as starting point for the new curve. * @throws CloneNotSupportedException Thrown, when the curve could not be cloned. */ public CurveBuilder(Curve curve) throws CloneNotSupportedException { this.curve = (Curve)curve.clone(); } /* (non-Javadoc) * @see net.finmath.marketdata.model.curves.CurveBuilderInterface#build() */ @Override public CurveInterface build() throws CloneNotSupportedException { Curve buildCurve = curve; curve = null; return buildCurve; } /** * Set the interpolation method of the curve. * * @param interpolationMethod The interpolation method of the curve. * @return A self reference to this curve build object. */ public CurveBuilderInterface setInterpolationMethod(InterpolationMethod interpolationMethod) { curve.interpolationMethod = interpolationMethod; return this; } /** * Set the extrapolation method of the curve. * * @param extrapolationMethod The extrapolation method of the curve. * @return A self reference to this curve build object. */ public CurveBuilderInterface setExtrapolationMethod(ExtrapolationMethod extrapolationMethod) { curve.extrapolationMethod = extrapolationMethod; return this; } /** * Set the interpolationEntity of the curve. * * @param interpolationEntity The interpolation entity of the curve. * @return A self reference to this curve build object. */ public CurveBuilderInterface setInterpolationEntity(InterpolationEntity interpolationEntity) { curve.interpolationEntity = interpolationEntity; return this; } /* (non-Javadoc) * @see net.finmath.marketdata.model.curves.CurveBuilderInterface#addPoint(double, double, boolean) */ @Override public CurveBuilderInterface addPoint(double time, double value, boolean isParameter) { curve.addPoint(time, value, isParameter); return this; } } private ArrayList points = new ArrayList(); private ArrayList pointsBeingParameters = new ArrayList(); private InterpolationMethod interpolationMethod = InterpolationMethod.CUBIC_SPLINE; private ExtrapolationMethod extrapolationMethod = ExtrapolationMethod.CONSTANT; private InterpolationEntity interpolationEntity = InterpolationEntity.LOG_OF_VALUE; private RationalFunctionInterpolation rationalFunctionInterpolation = null; private transient Object rationalFunctionInterpolationLazyInitLock = new Object(); private transient SoftReference> curveCacheReference = null; private static final long serialVersionUID = -4126228588123963885L; static NumberFormat formatterReal = NumberFormat.getInstance(Locale.US); /** * Create a curve with a given name, reference date and an interpolation method from given points * * @param name The name of this curve. * @param referenceDate The reference date for this curve, i.e., the date which defined t=0. * @param interpolationMethod The interpolation method used for the curve. * @param extrapolationMethod The extrapolation method used for the curve. * @param interpolationEntity The entity interpolated/extrapolated. * @param times A vector of times. * @param values A vector of corresponding values. */ public Curve(String name, LocalDate referenceDate, InterpolationMethod interpolationMethod, ExtrapolationMethod extrapolationMethod, InterpolationEntity interpolationEntity, double[] times, double[] values) { super(name, referenceDate); this.interpolationMethod = interpolationMethod; this.extrapolationMethod = extrapolationMethod; this.interpolationEntity = interpolationEntity; if(times.length != values.length) throw new IllegalArgumentException("Length of times not equal to length of values."); for(int i=0; i curveCache = curveCacheReference != null ? curveCacheReference.get() : null; if(curveCache == null) { curveCache = new ConcurrentHashMap(); curveCacheReference = new SoftReference>(curveCache); } Double valueFromCache = curveCache.get(time); if(valueFromCache != null) return valueFromCache.doubleValue(); double value = valueFromInterpolationEntity(getInterpolationEntityValue(time), time); curveCache.put(time, value); return value; } private double getInterpolationEntityValue(double time) { synchronized(rationalFunctionInterpolationLazyInitLock) { // Lazy initialization of interpolation function if(rationalFunctionInterpolation == null) { double[] pointsArray = new double[points.size()]; double[] valuesArray = new double[points.size()]; for(int i=0; ii in i = f(xi). * @param value The yi in i = f(xi). * @param isParameter If true, then this point is served via {@link #getParameter()} and changed via {@link #getCloneForParameter(double[])}, i.e., it can be calibrated. */ protected void addPoint(double time, double value, boolean isParameter) { synchronized (rationalFunctionInterpolationLazyInitLock) { if(interpolationEntity == InterpolationEntity.LOG_OF_VALUE_PER_TIME && time == 0) { if(value == 1.0 && isParameter == false) return; else throw new IllegalArgumentException("The interpolation method LOG_OF_VALUE_PER_TIME does not allow to add a value at time = 0 other than 1.0 (received " + value + ")."); } double interpolationEntityValue = interpolationEntityFromValue(value, time); int index = getTimeIndex(time); if(index >= 0) { if(points.get(index).value == interpolationEntityValue) return; // Already in list else if(isParameter) return; else throw new RuntimeException("Trying to add a value for a time for which another value already exists."); } else { // Insert the new point, retain ordering. Point point = new Point(time, interpolationEntityValue, isParameter); points.add(-index-1, point); if(isParameter) { // Add this point also to the list of parameters int parameterIndex = getParameterIndex(time); if(parameterIndex >= 0) new RuntimeException("Curve inconsistent."); pointsBeingParameters.add(-parameterIndex-1, point); } } this.rationalFunctionInterpolation = null; this.curveCacheReference = null; } } /** * Returns the interpolation method used by this curve. * * @return The interpolation method used by this curve. */ public InterpolationMethod getInterpolationMethod() { return interpolationMethod; } /** * Returns the extrapolation method used by this curve. * * @return The extrapolation method used by this curve. */ public ExtrapolationMethod getExtrapolationMethod() { return extrapolationMethod; } /** * Returns the interpolation entity used by this curve. * * @return The interpolation entity used by this curve. */ public InterpolationEntity getInterpolationEntity() { return interpolationEntity; } protected int getTimeIndex(double time) { Point point = new Point(time, Double.NaN, false); return java.util.Collections.binarySearch(points, point); } protected int getParameterIndex(double time) { Point point = new Point(time, Double.NaN, false); return java.util.Collections.binarySearch(pointsBeingParameters, point); } @Override public double[] getParameter() { double[] parameters = new double[pointsBeingParameters.size()]; for(int i=0; i(); newCurve.pointsBeingParameters = new ArrayList(); newCurve.rationalFunctionInterpolation = null; newCurve.curveCacheReference = null; for(Point point : points) { Point newPoint = (Point) point.clone(); newCurve.points.add(newPoint); if(point.isParameter) newCurve.pointsBeingParameters.add(newPoint); } return newCurve; } @Override public CurveInterface getCloneForParameter(double[] parameter) throws CloneNotSupportedException { if(Arrays.equals(parameter, getParameter())) return this; Curve newCurve = (Curve) this.clone(); newCurve.setParameterPrivate(parameter); return newCurve; } @Override public CurveBuilderInterface getCloneBuilder() throws CloneNotSupportedException { CurveBuilder curveBuilder = new CurveBuilder(this); return curveBuilder; } @Override public String toString() { /* * Pretty print curve (appended to standard toString) */ StringBuilder curveTableString = new StringBuilder(); NumberFormat formatTime = new DecimalFormat("0.00000000E0"); // Floating point time is accurate to 3+5 digits. for (Point point : points) { curveTableString.append(formatTime.format(point.time) + "\t"); curveTableString.append(FloatingpointDate.getDateFromFloatingPointDate(getReferenceDate(), point.time) + "\t"); curveTableString.append(valueFromInterpolationEntity(point.value, point.time) + "\n"); } return "Curve [points=" + points + ", pointsBeingParameters=" + pointsBeingParameters + ", interpolationMethod=" + interpolationMethod + ", extrapolationMethod=" + extrapolationMethod + ", interpolationEntity=" + interpolationEntity + ", rationalFunctionInterpolation=" + rationalFunctionInterpolation + ", toString()=" + super.toString() + ",\n" + curveTableString + "]"; } private void readObject(java.io.ObjectInputStream in) throws ClassNotFoundException, IOException { in.defaultReadObject(); // initialization of transients rationalFunctionInterpolationLazyInitLock = new Object(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy