- 新增日食、月食、本地可见性、中心线、半影区域、SVG 图示与沙罗周期信息 - 新增行星冲合、留、方照、物理星历、视直径、相位、亮肢角、轨道节点等计算 - 新增木星伽利略卫星位置、现象与接触事件计算 - 新增恒星星表、星座判定、自行修正与观测辅助能力 - 新增 coord、formula、orbit、sundial、lite/sun、lite/moon 等扩展包 - 完善农历年号、月相英文别名、视差角、大气质量、折射、日晷与双星计算 - 增加 NASA、JPL Horizons、IMCCE 等回归测试数据与基线测试 - 重构基础算法文件组织,补充大量公开 API 注释和语义回归测试 - 更新中文和英文 README,补充示例、精度说明、SVG 配图
100 lines
2.9 KiB
Go
100 lines
2.9 KiB
Go
package basic
|
|
|
|
import "math"
|
|
|
|
func eventFixedScanRefine(seed, halfWindow, step float64, fn func(float64) float64) float64 {
|
|
start := seed - halfWindow
|
|
bestJD := start
|
|
bestAbs := math.Abs(fn(start))
|
|
samples := int(math.Round((2 * halfWindow) / step))
|
|
for i := 1; i < samples; i++ {
|
|
candidateJD := start + float64(i)*step
|
|
candidateAbs := math.Abs(fn(candidateJD))
|
|
if candidateAbs < bestAbs {
|
|
bestAbs = candidateAbs
|
|
bestJD = candidateJD
|
|
}
|
|
}
|
|
return bestJD
|
|
}
|
|
|
|
func eventZeroBracket(leftJD, leftVal, centerJD, centerVal, rightJD, rightVal float64) (float64, float64, float64, float64, bool) {
|
|
if leftVal == 0 {
|
|
return leftJD, leftJD, leftVal, leftVal, true
|
|
}
|
|
if centerVal == 0 {
|
|
return centerJD, centerJD, centerVal, centerVal, true
|
|
}
|
|
if rightVal == 0 {
|
|
return rightJD, rightJD, rightVal, rightVal, true
|
|
}
|
|
if leftVal*centerVal < 0 {
|
|
return leftJD, centerJD, leftVal, centerVal, true
|
|
}
|
|
if centerVal*rightVal < 0 {
|
|
return centerJD, rightJD, centerVal, rightVal, true
|
|
}
|
|
if leftVal*rightVal < 0 {
|
|
return leftJD, rightJD, leftVal, rightVal, true
|
|
}
|
|
return 0, 0, 0, 0, false
|
|
}
|
|
|
|
// eventZeroRefine 细化 seed 附近的零点;若找不到可用括号区间,则退回旧的固定步长扫描。
|
|
// eventZeroRefine refines a nearby zero crossing and falls back to the legacy
|
|
// fixed-step scan when no usable bracket is found.
|
|
func eventZeroRefine(seed, halfWindow, step float64, fn func(float64) float64) float64 {
|
|
leftJD := seed - halfWindow
|
|
centerJD := seed
|
|
rightJD := seed + halfWindow
|
|
leftVal := fn(leftJD)
|
|
centerVal := fn(centerJD)
|
|
rightVal := fn(rightJD)
|
|
|
|
bestJD := centerJD
|
|
bestAbs := math.Abs(centerVal)
|
|
if candidateAbs := math.Abs(leftVal); candidateAbs < bestAbs {
|
|
bestAbs = candidateAbs
|
|
bestJD = leftJD
|
|
}
|
|
if candidateAbs := math.Abs(rightVal); candidateAbs < bestAbs {
|
|
bestAbs = candidateAbs
|
|
bestJD = rightJD
|
|
}
|
|
|
|
bracketLeftJD, bracketRightJD, bracketLeftVal, bracketRightVal, ok := eventZeroBracket(leftJD, leftVal, centerJD, centerVal, rightJD, rightVal)
|
|
if !ok {
|
|
return eventFixedScanRefine(seed, halfWindow, step, fn)
|
|
}
|
|
if bracketLeftJD == bracketRightJD {
|
|
return bracketLeftJD
|
|
}
|
|
|
|
for i := 0; i < 8; i++ {
|
|
candidateJD := (bracketLeftJD + bracketRightJD) / 2
|
|
if bracketRightVal != bracketLeftVal {
|
|
secantJD := bracketRightJD - bracketRightVal*(bracketRightJD-bracketLeftJD)/(bracketRightVal-bracketLeftVal)
|
|
if secantJD > bracketLeftJD && secantJD < bracketRightJD {
|
|
candidateJD = secantJD
|
|
}
|
|
}
|
|
candidateVal := fn(candidateJD)
|
|
candidateAbs := math.Abs(candidateVal)
|
|
if candidateAbs < bestAbs {
|
|
bestAbs = candidateAbs
|
|
bestJD = candidateJD
|
|
}
|
|
if candidateVal == 0 || math.Abs(bracketRightJD-bracketLeftJD) <= step {
|
|
break
|
|
}
|
|
if bracketLeftVal*candidateVal < 0 {
|
|
bracketRightJD = candidateJD
|
|
bracketRightVal = candidateVal
|
|
continue
|
|
}
|
|
bracketLeftJD = candidateJD
|
|
bracketLeftVal = candidateVal
|
|
}
|
|
return bestJD
|
|
}
|