astro/basic/orbit_kepler.go
starainrt 3ffdbe0034
feat: 扩展天文计算能力
- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息
- 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算
- 新增木星伽利略卫星位置、现象与接触事件计算
- 新增恒星星表、星座判定、自行修正与观测辅助能力
- 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包
- 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算
- 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试
- 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试
- 更新中文和英文 README,补充示例、精度说明、SVG 配图
2026-05-01 22:38:44 +08:00

239 lines
8.0 KiB
Go

package basic
import (
"math"
. "b612.me/astro/tools"
)
// OrbitMeanMotion 返回历元平均角速度,单位度/日。
// 对近日点形式的抛物/双曲轨道返回 NaN。
func OrbitMeanMotion(elements OrbitElements) float64 {
if elements.usesPerihelionForm() {
if !elements.validPerihelionForm() || elements.E >= 1 {
return math.NaN()
}
semiMajorAxis := elements.Q / (1 - elements.E)
if !isFinitePositive(semiMajorAxis) {
return math.NaN()
}
if isFinite(elements.MDot) && elements.MDot != 0 {
return elements.MDot
}
return gaussianGravitationalConstant / math.Pow(semiMajorAxis, 1.5) * deg
}
if !elements.validEllipticClassical() {
return math.NaN()
}
if isFinite(elements.MDot) && elements.MDot != 0 {
return elements.MDot
}
return gaussianGravitationalConstant / math.Pow(elements.A, 1.5) * deg
}
// OrbitMeanAnomaly 返回给定 TT/TDB 儒略日的平近点角,单位度。
// 对抛物/双曲轨道返回 NaN。
func OrbitMeanAnomaly(jd float64, elements OrbitElements) float64 {
meanAnomalyDeg, ok := orbitMeanAnomalyDegAt(jd, elements)
if !ok {
return math.NaN()
}
return Limit360(meanAnomalyDeg)
}
// OrbitEccentricAnomaly 返回给定 TT/TDB 儒略日的偏近点角,单位度。
// 仅适用于椭圆轨道;对抛物/双曲轨道返回 NaN。
func OrbitEccentricAnomaly(jd float64, elements OrbitElements) float64 {
resolved := orbitElementsAt(jd, elements)
meanAnomalyDeg, ok := orbitMeanAnomalyDegAt(jd, elements)
if !ok || !resolved.validEllipticClassical() && !(resolved.validPerihelionForm() && resolved.E < 1) {
return math.NaN()
}
eccentricAnomaly, ok := orbitEccentricAnomalyRad(meanAnomalyDeg*rad, resolved.E)
if !ok {
return math.NaN()
}
return Limit360(eccentricAnomaly * deg)
}
// OrbitTrueAnomaly 返回给定 TT/TDB 儒略日的真近点角,单位度。
func OrbitTrueAnomaly(jd float64, elements OrbitElements) float64 {
trueAnomaly, _, _, ok := orbitTrueAnomalyAndRadius(jd, elements)
if !ok {
return math.NaN()
}
return Limit360(trueAnomaly * deg)
}
func orbitMeanAnomalyDegAt(jd float64, elements OrbitElements) (float64, bool) {
if !isFinite(jd) {
return math.NaN(), false
}
if elements.usesPerihelionForm() {
if !elements.validPerihelionForm() || elements.E >= 1 {
return math.NaN(), false
}
semiMajorAxis := elements.Q / (1 - elements.E)
if !isFinitePositive(semiMajorAxis) {
return math.NaN(), false
}
meanMotion := elements.MDot
if !isFinite(meanMotion) || meanMotion == 0 {
meanMotion = gaussianGravitationalConstant / math.Pow(semiMajorAxis, 1.5) * deg
}
return meanMotion * (jd - elements.TpJD), true
}
resolved := orbitElementsAt(jd, elements)
if !resolved.validEllipticClassical() {
return math.NaN(), false
}
if isFinite(elements.MDot) && elements.MDot != 0 {
return elements.M0 + elements.MDot*(jd-elements.EpochJD), true
}
meanMotion := gaussianGravitationalConstant / math.Pow(resolved.A, 1.5) * deg
return resolved.M0 + meanMotion*(jd-elements.EpochJD), true
}
func orbitEccentricAnomalyRad(meanAnomalyRad, eccentricity float64) (float64, bool) {
if !isFinite(meanAnomalyRad) || !isFinite(eccentricity) || eccentricity < 0 || eccentricity >= 1 {
return math.NaN(), false
}
if meanAnomalyRad > math.Pi {
meanAnomalyRad -= 2 * math.Pi
} else if meanAnomalyRad < -math.Pi {
meanAnomalyRad += 2 * math.Pi
}
eccentricAnomaly := meanAnomalyRad
if eccentricity >= 0.8 {
eccentricAnomaly = math.Pi
if meanAnomalyRad < 0 {
eccentricAnomaly = -math.Pi
}
}
for i := 0; i < 32; i++ {
sinE, cosE := math.Sincos(eccentricAnomaly)
delta := (eccentricAnomaly - eccentricity*sinE - meanAnomalyRad) / (1 - eccentricity*cosE)
eccentricAnomaly -= delta
if math.Abs(delta) < 1e-14 {
return eccentricAnomaly, true
}
}
return eccentricAnomaly, true
}
func orbitHyperbolicAnomaly(meanAnomaly, eccentricity float64) (float64, bool) {
if !isFinite(meanAnomaly) || !isFinite(eccentricity) || eccentricity <= 1 {
return math.NaN(), false
}
hyperbolicAnomaly := math.Asinh(meanAnomaly / eccentricity)
if hyperbolicAnomaly == 0 && meanAnomaly != 0 {
hyperbolicAnomaly = math.Log(2*math.Abs(meanAnomaly)/eccentricity + 1.8)
if meanAnomaly < 0 {
hyperbolicAnomaly = -hyperbolicAnomaly
}
}
for i := 0; i < 32; i++ {
sinhH := math.Sinh(hyperbolicAnomaly)
coshH := math.Cosh(hyperbolicAnomaly)
delta := (eccentricity*sinhH - hyperbolicAnomaly - meanAnomaly) / (eccentricity*coshH - 1)
hyperbolicAnomaly -= delta
if math.Abs(delta) < 1e-14 {
return hyperbolicAnomaly, true
}
}
return hyperbolicAnomaly, true
}
func orbitTrueAnomalyAndRadius(jd float64, elements OrbitElements) (trueAnomaly, radius float64, resolved OrbitElements, ok bool) {
resolved = orbitElementsAt(jd, elements)
if resolved.usesPerihelionForm() {
if !resolved.validPerihelionForm() {
return math.NaN(), math.NaN(), resolved, false
}
switch {
case math.Abs(resolved.E-1) <= orbitParabolicTolerance:
return orbitParabolicTrueAnomalyAndRadius(jd, resolved)
case resolved.E < 1:
return orbitEllipticTrueAnomalyAndRadiusFromPerihelion(jd, resolved)
default:
return orbitHyperbolicTrueAnomalyAndRadius(jd, resolved)
}
}
if !resolved.validEllipticClassical() {
return math.NaN(), math.NaN(), resolved, false
}
meanAnomalyDeg, ok := orbitMeanAnomalyDegAt(jd, elements)
if !ok {
return math.NaN(), math.NaN(), resolved, false
}
trueAnomaly, radius, ok = orbitEllipticTrueAnomalyAndRadius(meanAnomalyDeg*rad, resolved.A, resolved.E)
return trueAnomaly, radius, resolved, ok
}
func orbitEllipticTrueAnomalyAndRadiusFromPerihelion(jd float64, elements OrbitElements) (trueAnomaly, radius float64, resolved OrbitElements, ok bool) {
semiMajorAxis := elements.Q / (1 - elements.E)
if !isFinitePositive(semiMajorAxis) {
return math.NaN(), math.NaN(), elements, false
}
meanAnomalyDeg, ok := orbitMeanAnomalyDegAt(jd, elements)
if !ok {
return math.NaN(), math.NaN(), elements, false
}
trueAnomaly, radius, ok = orbitEllipticTrueAnomalyAndRadius(meanAnomalyDeg*rad, semiMajorAxis, elements.E)
if !ok {
return math.NaN(), math.NaN(), elements, false
}
resolved = elements
resolved.A = semiMajorAxis
return trueAnomaly, radius, resolved, true
}
func orbitEllipticTrueAnomalyAndRadius(meanAnomalyRad, semiMajorAxis, eccentricity float64) (float64, float64, bool) {
eccentricAnomaly, ok := orbitEccentricAnomalyRad(meanAnomalyRad, eccentricity)
if !ok {
return math.NaN(), math.NaN(), false
}
sinE, cosE := math.Sincos(eccentricAnomaly)
radius := semiMajorAxis * (1 - eccentricity*cosE)
trueAnomaly := math.Atan2(math.Sqrt(1-eccentricity*eccentricity)*sinE, cosE-eccentricity)
return trueAnomaly, radius, true
}
func orbitParabolicTrueAnomalyAndRadius(jd float64, elements OrbitElements) (trueAnomaly, radius float64, resolved OrbitElements, ok bool) {
if !isFinitePositive(elements.Q) || !isFinite(elements.TpJD) {
return math.NaN(), math.NaN(), elements, false
}
w := 1.5 * gaussianGravitationalConstant * (jd - elements.TpJD) / (math.Sqrt2 * math.Pow(elements.Q, 1.5))
y := math.Cbrt(w + math.Sqrt(w*w+1))
if y == 0 {
return 0, elements.Q, elements, true
}
d := y - 1/y
trueAnomaly = 2 * math.Atan(d)
radius = elements.Q * (1 + d*d)
resolved = elements
return trueAnomaly, radius, resolved, true
}
func orbitHyperbolicTrueAnomalyAndRadius(jd float64, elements OrbitElements) (trueAnomaly, radius float64, resolved OrbitElements, ok bool) {
if !isFinitePositive(elements.Q) || !isFinite(elements.TpJD) || !isFinite(elements.E) || elements.E <= 1 {
return math.NaN(), math.NaN(), elements, false
}
semiMajorAxis := elements.Q / (elements.E - 1)
meanAnomaly := gaussianGravitationalConstant * (jd - elements.TpJD) / math.Pow(semiMajorAxis, 1.5)
hyperbolicAnomaly, ok := orbitHyperbolicAnomaly(meanAnomaly, elements.E)
if !ok {
return math.NaN(), math.NaN(), elements, false
}
radius = semiMajorAxis * (elements.E*math.Cosh(hyperbolicAnomaly) - 1)
trueAnomaly = 2 * math.Atan(math.Sqrt((elements.E+1)/(elements.E-1))*math.Tanh(hyperbolicAnomaly/2))
resolved = elements
resolved.A = -semiMajorAxis
return trueAnomaly, radius, resolved, true
}