400 lines
5.7 KiB
C++
Raw Normal View History

2013-09-29 14:16:03 +02:00
//
2013-09-29 14:13:36 +02:00
// FILE: Complex.cpp
// AUTHOR: Rob Tillaart
2022-10-30 11:11:07 +01:00
// VERSION: 0.3.2
2020-11-27 11:10:47 +01:00
// PURPOSE: Arduino library for Complex math
// URL: https://github.com/RobTillaart/Complex
// http://arduino.cc/playground/Main/ComplexMath
2013-09-29 14:13:36 +02:00
//
2022-10-30 11:11:07 +01:00
// HISTORY: see changelog.md
2021-09-14 11:58:53 +02:00
2013-09-29 14:16:03 +02:00
2020-11-27 11:10:47 +01:00
#include "Complex.h"
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2022-10-30 11:11:07 +01:00
// PRINTING
size_t Complex::printTo(Print& p) const
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
size_t n = 0;
n += p.print(re, 3);
n += p.print(' ');
n += p.print(im, 3);
n += p.print('i');
return n;
};
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2018-04-02 20:48:12 +02:00
void Complex::polar(const float modulus, const float phase)
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
re = modulus * cos(phase);
im = modulus * sin(phase);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::reciprocal() const
2013-09-29 14:13:36 +02:00
{
2021-01-29 12:31:58 +01:00
float f = 1.0 / (re * re + im * im);
float r = re * f;
float i = -im * f;
return Complex(r, i);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
//
2022-10-30 11:11:07 +01:00
// EQUALITIES
2021-09-14 11:58:53 +02:00
//
2021-11-16 09:37:28 +01:00
bool Complex::operator == (const Complex &c) const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (re == c.re) && (im == c.im);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
bool Complex::operator != (const Complex &c) const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (re != c.re) || (im != c.im);
2013-09-29 14:16:03 +02:00
}
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
//
2022-10-30 11:11:07 +01:00
// NEGATE
2021-09-14 11:58:53 +02:00
//
2021-11-16 09:37:28 +01:00
Complex Complex::operator - () const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return Complex(-re, -im);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
//
2022-10-30 11:11:07 +01:00
// BASIC MATH
2021-09-14 11:58:53 +02:00
//
2021-11-16 09:37:28 +01:00
Complex Complex::operator + (const Complex &c) const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return Complex(re + c.re, im + c.im);
2013-09-29 14:16:03 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::operator - (const Complex &c) const
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
return Complex(re - c.re, im - c.im);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::operator * (const Complex &c) const
2013-09-29 14:13:36 +02:00
{
2018-04-02 20:48:12 +02:00
float r = re * c.re - im * c.im;
float i = re * c.im + im * c.re;
2018-01-29 09:28:23 +01:00
return Complex(r, i);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::operator / (const Complex &c) const
2013-09-29 14:13:36 +02:00
{
2021-01-29 12:31:58 +01:00
float f = 1.0/(c.re * c.re + c.im * c.im);
2018-04-02 20:48:12 +02:00
float r = (re * c.re + im * c.im) * f;
float i = (im * c.re - re * c.im) * f;
2018-01-29 09:28:23 +01:00
return Complex(r, i);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
Complex& Complex::operator += (const Complex &c)
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
re += c.re;
im += c.im;
return *this;
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
Complex& Complex::operator -= (const Complex &c)
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
re -= c.re;
im -= c.im;
return *this;
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
Complex& Complex::operator *= (const Complex &c)
2013-09-29 14:13:36 +02:00
{
2018-04-02 20:48:12 +02:00
float r = re * c.re - im * c.im;
float i = re * c.im + im * c.re;
2018-01-29 09:28:23 +01:00
re = r;
im = i;
return *this;
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
Complex& Complex::operator /= (const Complex &c)
2013-09-29 14:13:36 +02:00
{
2021-01-29 12:31:58 +01:00
float f = 1.0/(c.re * c.re + c.im * c.im);
2018-04-02 20:48:12 +02:00
float r = (re * c.re + im * c.im) * f;
float i = (im * c.re - re * c.im) * f;
2018-01-29 09:28:23 +01:00
re = r;
im = i;
return *this;
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2013-09-29 14:16:03 +02:00
//
2022-10-30 11:11:07 +01:00
// POWER FUNCTIONS
2013-09-29 14:16:03 +02:00
//
2021-11-16 09:37:28 +01:00
Complex Complex::c_sqr() const
{
2018-04-02 20:48:12 +02:00
float r = re * re - im * im;
float i = 2 * re * im;
2018-01-29 09:28:23 +01:00
return Complex(r, i);
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_sqrt() const
2013-09-29 14:13:36 +02:00
{
2018-04-02 20:48:12 +02:00
float m = modulus();
2021-01-29 12:31:58 +01:00
float r = sqrt(0.5 * (m + re));
float i = sqrt(0.5 * (m - re));
2018-01-29 09:28:23 +01:00
if (im < 0) i = -i;
return Complex(r, i);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_exp() const
2013-09-29 14:13:36 +02:00
{
2018-04-02 20:48:12 +02:00
float e = exp(re);
2018-01-29 09:28:23 +01:00
return Complex(e * cos(im), e * sin(im));
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_log() const
2013-09-29 14:13:36 +02:00
{
2018-04-02 20:48:12 +02:00
float m = modulus();
float p = phase();
2021-01-29 12:31:58 +01:00
if (p > PI) p -= 2 * PI;
2018-01-29 09:28:23 +01:00
return Complex(log(m), p);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_pow(const Complex &c) const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
Complex t = c_log();
t = t * c;
return t.c_exp();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_logn(const Complex &c) const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
Complex t = c;
return c_log()/t.c_log();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_log10() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return c_logn(10);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2013-09-29 14:16:03 +02:00
//
2022-10-30 11:11:07 +01:00
// GONIO I - SIN COS TAN
2013-09-29 14:16:03 +02:00
//
2021-11-16 09:37:28 +01:00
Complex Complex::c_sin() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return Complex(sin(re) * cosh(im), cos(re) * sinh(im));
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_cos() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return Complex(cos(re) * cosh(im), -sin(re) * sinh(im));
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_tan() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
/* faster but 350 bytes longer!!
2018-04-02 20:48:12 +02:00
float s = sin(re);
float c = cos(re);
float sh = sinh(im);
float ch = cosh(im);
// return Complex(s*ch, c*sh) / Complex(c*ch, -s*sh);
2018-04-02 20:48:12 +02:00
float r0 = s*ch;
float i0 = c*sh;
float cre = c*ch;
float cim = -s*sh;
float f = 1.0/(cre*cre + cim*cim);
float r = r0 * cre + i0 * cim;
float i = r0 * cim - i0 * cre;
return Complex(r * f, -i * f);
*/
2018-01-29 09:28:23 +01:00
return c_sin() / c_cos();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::gonioHelper1(const byte mode) const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
Complex c = (one - this->c_sqr()).c_sqrt();
if (mode == 0)
{
c = c + *this * Complex(0,-1);
}
else
{
c = *this + c * Complex(0,-1);
}
c = c.c_log() * Complex(0,1);
return c;
2013-09-29 14:16:03 +02:00
}
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_asin() const
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
return gonioHelper1(0);
2013-09-29 14:16:03 +02:00
}
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_acos() const
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
return gonioHelper1(1);
2013-09-29 14:16:03 +02:00
}
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_atan() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (Complex(0,-1) * (Complex(re, im - 1)/Complex(-re, -im - 1)).c_log()) * 0.5;
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2013-09-29 14:16:03 +02:00
//
2022-10-30 11:11:07 +01:00
// GONIO II - CSC SEC COT
2013-09-29 14:16:03 +02:00
//
2021-11-16 09:37:28 +01:00
Complex Complex::c_csc() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return one / c_sin();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_sec() const
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
return one / c_cos();
2013-09-29 14:16:03 +02:00
}
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_cot() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return one / c_tan();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_acsc() const
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
return (one / *this).c_asin();
2013-09-29 14:16:03 +02:00
}
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_asec() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (one / *this).c_acos();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_acot() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (one / *this).c_atan();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2013-09-29 14:16:03 +02:00
//
2022-10-30 11:11:07 +01:00
// GONIO HYPERBOLICUS I
2013-09-29 14:16:03 +02:00
//
2021-11-16 09:37:28 +01:00
Complex Complex::c_sinh() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return Complex(cos(im) * sinh(re), sin(im) * cosh(re));
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_cosh() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return Complex(cos(im) * cosh(re), sin(im) * sinh(re));
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_tanh() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return c_sinh() / c_cosh();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::gonioHelper2(const byte mode) const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
Complex c = c_sqr();
if (mode == 0)
{
c += 1;
}
else
{
c -= 1;
}
c = (*this + c.c_sqrt()).c_log();
return c;
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_asinh() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return gonioHelper2(0);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_acosh() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return gonioHelper2(1);
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_atanh() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
Complex c = (*this + one).c_log();
c = c - (-(*this - one)).c_log();
return c * 0.5;
2013-09-29 14:16:03 +02:00
}
2013-09-29 14:13:36 +02:00
2021-09-14 11:58:53 +02:00
2013-09-29 14:16:03 +02:00
//
2022-10-30 11:11:07 +01:00
// GONIO HYPERBOLICUS II
2013-09-29 14:16:03 +02:00
//
2021-11-16 09:37:28 +01:00
Complex Complex::c_csch() const
2013-09-29 14:16:03 +02:00
{
2018-01-29 09:28:23 +01:00
return one / c_sinh();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_sech() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return one / c_cosh();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_coth() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return one / c_tanh();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_acsch() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (one / *this).c_asinh();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_asech() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (one / *this).c_acosh();
2013-09-29 14:13:36 +02:00
}
2021-09-14 11:58:53 +02:00
2021-11-16 09:37:28 +01:00
Complex Complex::c_acoth() const
2013-09-29 14:13:36 +02:00
{
2018-01-29 09:28:23 +01:00
return (one / *this).c_atanh();
2013-09-29 14:16:03 +02:00
}
2021-12-14 15:48:21 +01:00
2021-09-14 11:58:53 +02:00
// --- END OF FILE ---
2021-12-14 15:48:21 +01:00