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

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

/*
 * (c) Copyright Christian P. Fries, Germany. Contact: [email protected].
 *
 * Created on 20.05.2005
 */
package net.finmath.marketdata.model.curves;

import java.io.Serializable;

import net.finmath.marketdata.model.AnalyticModelInterface;

/**
 * A discount curve derived from a given forward curve.
 * 
 * The discount factors df(t) are defined at t = k * d
 * for integers k via
 * df(t+d) = df(t) / (1 + f(t) * d) and
 * for t = k * d and 0 < r < d
 * via df(t+r) = df(t) / (1 + f(t) * r)
 * where d is a given the payment offset and f(t) is the forward curve.
 * 
 * 

* Note that a special interpolation is performed for in-between points. * Hence, creating a {@link ForwardCurveFromDiscountCurve} and from it * a DiscountCurveFromForwardCurve will not recover the original curve * since interpolation points may be lost. * *

* * @author Christian Fries */ public class DiscountCurveFromForwardCurve extends AbstractCurve implements Serializable, DiscountCurveInterface { private static final long serialVersionUID = -4126228588123963885L; private String forwardCurveName; private ForwardCurveInterface forwardCurve; private final double timeScaling; /** * Create a discount curve using a given forward curve. * The discount factors df(t) are defined at t = k * d for integers k * via df(t+d) = df(t) / (1 + f(t) * d) and * for t = k * d and 0 < r < d * via df(t+r) = df(t) / (1 + f(t) * r) * where d is a given the payment offset and f(t) is the forward curve. * * @param forwardCurveName The name of the forward curve used for calculation of the discount factors. * @param periodLengthTimeScaling A scaling factor applied to d, adjusting for the internal double time to the period length daycount fraction (note that this may only be an approximate solution to capture daycount effects). */ public DiscountCurveFromForwardCurve(String forwardCurveName, double periodLengthTimeScaling) { super("DiscountCurveFromForwardCurve(" + forwardCurveName + ")", null); this.forwardCurveName = forwardCurveName; this.timeScaling = periodLengthTimeScaling; } /** * Create a discount curve using a given forward curve. * The discount factors df(t) are defined at t = k * d for integers k * via df(t+d) = df(t) / (1 + f(t) * d) and * for t = k * d and 0 < r < d * via df(t+r) = df(t) / (1 + f(t) * r) * where d is a given the payment offset and f(t) is the forward curve. * * @param forwardCurve The forward curve used for calculation of the discount factors. * @param periodLengthTimeScaling A scaling factor applied to d, adjusting for the internal double time to the period length daycount fraction (note that this may only be an approximate solution to capture daycount effects). */ public DiscountCurveFromForwardCurve(ForwardCurveInterface forwardCurve, double periodLengthTimeScaling) { super("DiscountCurveFromForwardCurve" + forwardCurve.getName() + ")", null); this.forwardCurve = forwardCurve; this.timeScaling = periodLengthTimeScaling; } /** * Create a discount curve using a given forward curve. * The discount factors df(t) are defined at t = k * d for integers k * via df(t+d) = df(t) / (1 + f(t) * d) and * for t = k * d and 0 < r < d * via df(t+r) = df(t) / (1 + f(t) * r) * where d is a given the payment offset and f(t) is the forward curve. * * @param forwardCurveName The name of the forward curve used for calculation of the discount factors. */ public DiscountCurveFromForwardCurve(String forwardCurveName) { this(forwardCurveName, 1.0); } /** * Create a discount curve using a given forward curve. * The discount factors df(t) are defined at t = k * d for integers k * via df(t+d) = df(t) / (1 + f(t) * d) and * for t = k * d and 0 < r < d * via df(t+r) = df(t) / (1 + f(t) * r) * where d is a given the payment offset and f(t) is the forward curve. * * @param forwardCurve The forward curve used for calculation of the discount factors. */ public DiscountCurveFromForwardCurve(ForwardCurveInterface forwardCurve) { this(forwardCurve, 1.0); } /* (non-Javadoc) * @see net.finmath.marketdata.DiscountCurveInterface#getDiscountFactor(double) */ @Override public double getDiscountFactor(double maturity) { return getDiscountFactor(null, maturity); } /* (non-Javadoc) * @see net.finmath.marketdata.DiscountCurveInterface#getDiscountFactor(double) */ @Override public double getDiscountFactor(AnalyticModelInterface model, double maturity) { ForwardCurveInterface forwardCurve; if(this.forwardCurve != null) forwardCurve = this.forwardCurve; else forwardCurve = model.getForwardCurve(forwardCurveName); if(forwardCurve == null) throw new IllegalArgumentException("No forward curve given and no forward curve found in the model under the name " + forwardCurveName + "."); double time = 0; double discountFactor = 1.0; double paymentOffset = 0; while(time < maturity) { paymentOffset = forwardCurve.getPaymentOffset(time); if(paymentOffset <= 0) throw new RuntimeException("Trying to calculate a discount curve from a forward curve with non-positive payment offset."); discountFactor /= 1.0 + forwardCurve.getForward(model, time) * Math.min(paymentOffset, maturity-time) * timeScaling; time += paymentOffset; } return discountFactor; } /* (non-Javadoc) * @see net.finmath.marketdata.model.curves.CurveInterface#getValue(double) */ @Override public double getValue(AnalyticModelInterface model, double time) { return getDiscountFactor(model, time); } @Override public double[] getParameter() { return null; } @Override public void setParameter(double[] parameter) { } @Override public CurveBuilderInterface getCloneBuilder() throws CloneNotSupportedException { throw new CloneNotSupportedException(); } /* (non-Javadoc) * @see java.lang.Object#hashCode() */ @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((forwardCurve == null) ? 0 : forwardCurve.hashCode()); result = prime * result + ((forwardCurveName == null) ? 0 : forwardCurveName.hashCode()); long temp; temp = Double.doubleToLongBits(timeScaling); result = prime * result + (int) (temp ^ (temp >>> 32)); return result; } /* (non-Javadoc) * @see java.lang.Object#equals(java.lang.Object) */ @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; DiscountCurveFromForwardCurve other = (DiscountCurveFromForwardCurve) obj; if (forwardCurve == null) { if (other.forwardCurve != null) return false; } else if (!forwardCurve.equals(other.forwardCurve)) return false; if (forwardCurveName == null) { if (other.forwardCurveName != null) return false; } else if (!forwardCurveName.equals(other.forwardCurveName)) return false; if (Double.doubleToLongBits(timeScaling) != Double .doubleToLongBits(other.timeScaling)) return false; return true; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy