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

net.finmath.fouriermethod.models.BatesModel Maven / Gradle / Ivy

/*
 * (c) Copyright Christian P. Fries, Germany. Contact: [email protected].
 *
 * Created on 23.03.2014
 */
package net.finmath.fouriermethod.models;

import org.apache.commons.math3.complex.Complex;

import net.finmath.fouriermethod.CharacteristicFunction;

/**
 * Implements the characteristic function of a Bates model.
 *
 * The Bates model for an underlying \( S \) is given by
 * \[
 * 	dS(t) = r^{\text{c}} S(t) dt + \sqrt{V(t)} S(t) dW_{1}(t) + S dJ, \quad S(0) = S_{0},
 * \]
 * \[
 * 	dV(t) = \kappa ( \theta - V(t) ) dt + \xi \sqrt{V(t)} dW_{2}(t), \quad V(0) = \sigma^2,
 * \]
 * \[
 * 	dW_{1} dW_{2} = \rho dt
 * \]
 * \[
 * 	dN(t) = r^{\text{d}} N(t) dt, \quad N(0) = N_{0},
 * \]
 * where \( W \) is Brownian motion and \( J \)  is a jump process (compound Poisson process).
 *
 * The free parameters of this model are:
 * 
*
\( S_{0} \)
spot - initial value of S
*
\( r \)
the risk free rate
*
\( \sigma \)
the initial volatility level
*
\( \xi \)
the volatility of volatility
*
\( \theta \)
the mean reversion level of the stochastic volatility
*
\( \kappa \)
the mean reversion speed of the stochastic volatility
*
\( \rho \)
the correlation of the Brownian drivers
*
\( a \)
the jump size mean
*
\( b \)
the jump size standard deviation
*
* * The process \( J \) is given by \( J(t) = \sum_{i=1}^{N(t)} (Y_{i}-1) \), where * \( \log(Y_{i}) \) are i.i.d. normals with mean \( a - \frac{1}{2} b^{2} \) and standard deviation \( b \). * Here \( a \) is the jump size mean and \( b \) is the jump size std. dev. * * The model can be rewritten as \( S = \exp(X) \), where * \[ * dX = \mu dt + \sqrt{V(t)} dW + dJ^{X}, \quad X(0) = \log(S_{0}), * \] * with * \[ * J^{X}(t) = \sum_{i=1}^{N(t)} \log(Y_{i}) * \] * with \( \mu = r - \frac{1}{2} \sigma^2 - (exp(a)-1) \lambda \). * * * @author Christian Fries * @author Andy Graf * @author Lorenzo Toricelli * @version 1.0 */ public class BatesModel implements CharacteristicFunctionModel { private final double initialValue; private final double riskFreeRate; // Actually the same as the drift (which is not stochastic) private final double[] volatility; private final double discountRate; private final double[] alpha; private final double[] beta; private final double[] sigma; private final double[] rho; private final double[] lambda; //3 constants private final double k; private final double delta; private final int numberOfFactors; /** * Create a two factor Bates model. * * @param initialValue Initial value of S. * @param riskFreeRate Risk free rate. * @param volatility Square root of initial value of the stochastic variance process V. * @param discountRate Rate used for the discount factor. * @param alpha The parameter alpha/beta is the mean reversion level of the variance process V. * @param beta Mean reversion speed of variance process V. * @param sigma Volatility of volatility. * @param rho Correlations of the Brownian drives (underlying, variance). * @param lambda Coefficients of for the jump intensity. * @param k Jump size mean. * @param delta Jump size variance. */ public BatesModel( double initialValue, double riskFreeRate, double[] volatility, double discountRate, double[] alpha, double[] beta, double[] sigma, double[] rho, double[] lambda, double k, double delta ) { super(); this.initialValue = initialValue; this.riskFreeRate = riskFreeRate; this.volatility = volatility; this.discountRate = discountRate; this.alpha = alpha; this.beta = beta; this.sigma = sigma; this.rho = rho; this.lambda = lambda; this.k = k; this.delta = delta; numberOfFactors = alpha.length; } /** * Create a two factor Bates model. * * @param initialValue Initial value of S. * @param riskFreeRate Risk free rate. * @param volatility Square root of initial value of the stochastic variance process V. * @param alpha The parameter alpha/beta is the mean reversion level of the variance process V. * @param beta Mean reversion speed of variance process V. * @param sigma Volatility of volatility. * @param rho Correlations of the Brownian drives (underlying, variance). * @param lambda Coefficients of for the jump intensity. * @param k Jump size mean. * @param delta Jump size variance. */ public BatesModel( double initialValue, double riskFreeRate, double[] volatility, double[] alpha, double[] beta, double[] sigma, double[] rho, double[] lambda, double k, double delta ) { this( initialValue, riskFreeRate, volatility, riskFreeRate, alpha, beta, sigma, rho, lambda, k, delta ); } /** * Create a one factor Bates model. * * @param initialValue Initial value of S. * @param riskFreeRate Risk free rate. * @param volatility Square root of initial value of the stochastic variance process V. * @param alpha The parameter alpha/beta is the mean reversion level of the variance process V. * @param beta Mean reversion speed of variance process V. * @param sigma Volatility of volatility. * @param rho Correlations of the Brownian drives (underlying, variance). * @param lambdaZero Constant part of the jump intensity. * @param lambdaOne Coefficients of the jump intensity, linear in variance. * @param k Jump size mean. * @param delta Jump size variance. */ public BatesModel( double initialValue, double riskFreeRate, double volatility, double alpha, double beta, double sigma, double rho, double lambdaZero, double lambdaOne, double k, double delta ) { this(initialValue, riskFreeRate, new double[]{ volatility }, new double[]{ alpha }, new double[]{ beta }, new double[]{ sigma }, new double[]{ rho }, new double[]{ lambdaZero, lambdaOne }, k, delta ); } /* (non-Javadoc) * @see net.finmath.fouriermethod.models.ProcessCharacteristicFunctionInterface#apply(double) */ @Override public CharacteristicFunction apply(final double time) { return new CharacteristicFunction() { @Override public Complex apply(Complex argument) { Complex iargument = argument.multiply(Complex.I); Complex c = iargument .multiply(iargument) .add(iargument.multiply(-1)) .multiply(0.5*delta*delta) .exp() .multiply(new Complex(1+k).pow(iargument)) .add(-1) .add(iargument.multiply(-k)); Complex[] gamma = new Complex[numberOfFactors]; Complex[] a = new Complex[numberOfFactors]; Complex[] b = new Complex[numberOfFactors]; for(int i = 0; i < numberOfFactors; i++) { gamma[i] = iargument .multiply(rho[i]*sigma[i]) .subtract(beta[i]) .pow(2) .subtract( iargument.multiply(iargument) .add(iargument.multiply(-1)) .multiply(0.5) .add(c.multiply(lambda[i+1])) .multiply(2*sigma[i]*sigma[i]) ) .sqrt() ; a[i] = iargument .multiply(rho[i] * sigma[i]) .subtract(beta[i]) .subtract(gamma[i]) .multiply((-alpha[i]*time)/(sigma[i]*sigma[i])) .subtract(iargument .multiply(rho[i]*sigma[i]) .subtract(beta[i]) .subtract(gamma[i]) .multiply(new Complex(1).divide(gamma[i].multiply(time).exp()) .subtract(1) .divide(gamma[i]) ) .multiply(0.5) .add(new Complex(1).divide(gamma[i].multiply(time).exp())) .log() .add(gamma[i].multiply(time)) .multiply((2*alpha[i])/(sigma[i]*sigma[i])) ) ; b[i] = iargument .multiply(iargument) .add(iargument.multiply(-1)) .multiply(0.5) .add(c.multiply(lambda[i+1])) .multiply(-2) .divide(iargument .multiply(rho[i] * sigma[i]) .subtract(beta[i]) .add(gamma[i] .multiply(new Complex(1).divide(gamma[i].multiply(time).exp()) .add(1) .divide(new Complex(1).divide(gamma[i].multiply(time).exp()) .subtract(1) ) ) ) ) ; } Complex characteristicFunction = a[0] .add(b[0].multiply(volatility[0])) .add(c.multiply(time*lambda[0])) .add(iargument.multiply(Math.log(initialValue)+time*riskFreeRate)) .add(-discountRate*time); if(numberOfFactors == 2) { characteristicFunction = characteristicFunction .add(a[1]) .add(b[1].multiply(volatility[1])); } characteristicFunction = characteristicFunction.exp(); return characteristicFunction; } }; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy