2008-04-05 10:25:40 -04:00
|
|
|
module CMath
|
|
|
|
|
|
|
|
include Math
|
|
|
|
|
|
|
|
alias exp! exp
|
|
|
|
alias log! log
|
|
|
|
alias log10! log10
|
|
|
|
alias sqrt! sqrt
|
|
|
|
|
|
|
|
alias sin! sin
|
|
|
|
alias cos! cos
|
|
|
|
alias tan! tan
|
|
|
|
|
|
|
|
alias sinh! sinh
|
|
|
|
alias cosh! cosh
|
|
|
|
alias tanh! tanh
|
|
|
|
|
|
|
|
alias asin! asin
|
|
|
|
alias acos! acos
|
|
|
|
alias atan! atan
|
|
|
|
alias atan2! atan2
|
|
|
|
|
|
|
|
alias asinh! asinh
|
|
|
|
alias acosh! acosh
|
|
|
|
alias atanh! atanh
|
|
|
|
|
|
|
|
def exp(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
exp!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(exp!(z.real) * cos!(z.imag),
|
|
|
|
exp!(z.real) * sin!(z.imag))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def log(*args)
|
|
|
|
z, b = args
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real? and z >= 0 and (b.nil? or b >= 0)
|
2008-04-05 10:25:40 -04:00
|
|
|
log!(*args)
|
|
|
|
else
|
|
|
|
r, theta = z.polar
|
|
|
|
a = Complex(log!(r.abs), theta)
|
|
|
|
if b
|
|
|
|
a /= log(b)
|
|
|
|
end
|
|
|
|
a
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def log10(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
log10!(z)
|
|
|
|
else
|
|
|
|
log(z) / log!(10)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def sqrt(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
if z >= 0
|
|
|
|
sqrt!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(0, sqrt!(-z))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
if z.imag < 0
|
2008-04-05 10:25:40 -04:00
|
|
|
sqrt(z.conjugate).conjugate
|
|
|
|
else
|
|
|
|
r = z.abs
|
|
|
|
x = z.real
|
|
|
|
Complex(sqrt!((r + x) / 2), sqrt!((r - x) / 2))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def sin(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
sin!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(sin!(z.real) * cosh!(z.imag),
|
|
|
|
cos!(z.real) * sinh!(z.imag))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def cos(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
cos!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(cos!(z.real) * cosh!(z.imag),
|
|
|
|
-sin!(z.real) * sinh!(z.imag))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def tan(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
tan!(z)
|
|
|
|
else
|
|
|
|
sin(z)/cos(z)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def sinh(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
sinh!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(sinh!(z.real) * cos!(z.imag),
|
|
|
|
cosh!(z.real) * sin!(z.imag))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def cosh(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
cosh!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(cosh!(z.real) * cos!(z.imag),
|
|
|
|
sinh!(z.real) * sin!(z.imag))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def tanh(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
tanh!(z)
|
|
|
|
else
|
|
|
|
sinh(z) / cosh(z)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def asin(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real? and z >= -1 and z <= 1
|
2008-04-05 10:25:40 -04:00
|
|
|
asin!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(0, -1.0) * log(Complex(0, 1.0) * z + sqrt(1.0 - z * z))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def acos(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real? and z >= -1 and z <= 1
|
2008-04-05 10:25:40 -04:00
|
|
|
acos!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(0, -1.0) * log(z + Complex(0, 1.0) * sqrt(1.0 - z * z))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def atan(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
atan!(z)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(0, 1.0) * log((Complex(0, 1.0) + z) / (Complex(0, 1.0) - z)) / 2.0
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def atan2(y,x)
|
2008-09-16 18:04:19 -04:00
|
|
|
if y.real? and x.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
atan2!(y,x)
|
|
|
|
else
|
2008-09-20 18:49:56 -04:00
|
|
|
Complex(0, -1.0) * log((x + Complex(0, 1.0) * y) / sqrt(x * x + y * y))
|
2008-04-05 10:25:40 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def acosh(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real? and z >= 1
|
2008-04-05 10:25:40 -04:00
|
|
|
acosh!(z)
|
|
|
|
else
|
|
|
|
log(z + sqrt(z * z - 1.0))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def asinh(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real?
|
2008-04-05 10:25:40 -04:00
|
|
|
asinh!(z)
|
|
|
|
else
|
|
|
|
log(z + sqrt(1.0 + z * z))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def atanh(z)
|
2008-09-16 18:04:19 -04:00
|
|
|
if z.real? and z >= -1 and z <= 1
|
2008-04-05 10:25:40 -04:00
|
|
|
atanh!(z)
|
|
|
|
else
|
|
|
|
log((1.0 + z) / (1.0 - z)) / 2.0
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
module_function :exp!
|
|
|
|
module_function :exp
|
|
|
|
module_function :log!
|
|
|
|
module_function :log
|
|
|
|
module_function :log10!
|
|
|
|
module_function :log10
|
|
|
|
module_function :sqrt!
|
|
|
|
module_function :sqrt
|
|
|
|
|
|
|
|
module_function :sin!
|
|
|
|
module_function :sin
|
|
|
|
module_function :cos!
|
|
|
|
module_function :cos
|
|
|
|
module_function :tan!
|
|
|
|
module_function :tan
|
|
|
|
|
|
|
|
module_function :sinh!
|
|
|
|
module_function :sinh
|
|
|
|
module_function :cosh!
|
|
|
|
module_function :cosh
|
|
|
|
module_function :tanh!
|
|
|
|
module_function :tanh
|
|
|
|
|
|
|
|
module_function :asin!
|
|
|
|
module_function :asin
|
|
|
|
module_function :acos!
|
|
|
|
module_function :acos
|
|
|
|
module_function :atan!
|
|
|
|
module_function :atan
|
|
|
|
module_function :atan2!
|
|
|
|
module_function :atan2
|
|
|
|
|
|
|
|
module_function :asinh!
|
|
|
|
module_function :asinh
|
|
|
|
module_function :acosh!
|
|
|
|
module_function :acosh
|
|
|
|
module_function :atanh!
|
|
|
|
module_function :atanh
|
|
|
|
|
|
|
|
end
|