001/* 002 * This file is part of McIDAS-V 003 * 004 * Copyright 2007-2025 005 * Space Science and Engineering Center (SSEC) 006 * University of Wisconsin - Madison 007 * 1225 W. Dayton Street, Madison, WI 53706, USA 008 * https://www.ssec.wisc.edu/mcidas/ 009 * 010 * All Rights Reserved 011 * 012 * McIDAS-V is built on Unidata's IDV and SSEC's VisAD libraries, and 013 * some McIDAS-V source code is based on IDV and VisAD source code. 014 * 015 * McIDAS-V is free software; you can redistribute it and/or modify 016 * it under the terms of the GNU Lesser Public License as published by 017 * the Free Software Foundation; either version 3 of the License, or 018 * (at your option) any later version. 019 * 020 * McIDAS-V is distributed in the hope that it will be useful, 021 * but WITHOUT ANY WARRANTY; without even the implied warranty of 022 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 023 * GNU Lesser Public License for more details. 024 * 025 * You should have received a copy of the GNU Lesser Public License 026 * along with this program. If not, see https://www.gnu.org/licenses/. 027 */ 028package edu.wisc.ssec.mcidasv.util; 029 030import java.util.Date; 031import java.util.Locale; 032import java.util.TimeZone; 033import java.text.DateFormat; 034import java.text.ParseException; 035import java.text.SimpleDateFormat; 036import java.io.PrintStream; 037 038 039/** 040 * Approximations de quelques calculs astronomiques relatifs aux calendriers terrestres. 041 * Les différents cycles astronomiques (notamment le jour, le mois et l'année) ne sont pas 042 * constants. Par exemple, la longueur de l'année tropicale (le nombre moyen de jours entre 043 * deux équinoxes vernales) était d'environ 365,242196 jours en 1900 et devrait être d'environ 044 * 365,242184 jours en 2100, soit un changement d'environ 1 seconde. Cette classe permet de 045 * calculer la longueur d'une année ou d'un mois à une date spécifiée. Toutefois, il est 046 * important de noter que les intervalles de temps calculés par les méthodes de cette classe 047 * sont des <strong>moyennes</strong>. Pour une année en particulier, l'intervalle de temps 048 * d'un équinoxe vernale au prochain peut s'écarter de cette moyenne de plusieurs minutes. 049 * <p> 050 * Les calculs de la longueur de l'année tropicale sont basés sur les travaux de Laskar (1986). 051 * Les calculs de la longueur des mois synodiques sont basés sur les travaux de Chapront-Touze et 052 * Chapront (1988).On peut lire plus de détails au sujet des calendrier terrestre au site 053 * <a href="http://webexhibits.org/calendars/year-astronomy.html">http://webexhibits.org/calendars/year-astronomy.html</a> 054 * ainsi que 055 * <a href="http://www.treasure-troves.com/astro/TropicalYear.html">http://www.treasure-troves.com/astro/TropicalYear.html</a>. 056 * 057 * @author Martin Desruisseaux (IRD) 058 * @version 3.00 059 * 060 * @since 2.0 061 * @module 062 */ 063public final class Calendar { 064 /** 065 * Nombre de millisecondes dans une journée. Cette constante est 066 * utilisée pour convertir des intervalles de temps du Java en 067 * nombre de jours. 068 */ 069 private static final double MILLIS_IN_DAY = 1000*60*60*24; 070 071 /** 072 * Jour julien correspondant à l'époch du Java (1er janvier 1970 à minuit). 073 * Cette constante est utilisée pour convertir des dates du Java en jour 074 * julien. 075 * 076 * La valeur {@link #julianDay} du 1er janvier 2000 00:00 GMT est 2451544.5 jours. 077 * La valeur {@link Date#getTime} du 1er janvier 2000 00:00 GMT est 10957 jours. 078 */ 079 private static final double JULIAN_DAY_1970 = 2451544.5-10957; 080 081 /** 082 * Do not allow instantiation of this class. 083 */ 084 private Calendar() { 085 } 086 087 /** 088 * Returns the julian day of the given date. This is the number of days elapsed since 089 * January 1st, 4713 before J.C. at noon GMT. This is named after <cite>Julius Scaliger</cite>, 090 * not to be confused to the number of days elapsed since the beginning of the year (named after 091 * <cite>Julius Caesar</cite>). 092 * 093 * @param time The time for which to evaluate the julian day. 094 * @return Number of days elapsed since January 1st, 4713 before J.C. at noon GMT. 095 */ 096 public static double julianDay(final Date time) { 097 return julianDay(time.getTime()); 098 } 099 100 /** 101 * Computes the {@linkplain #julianDay(Date) julian day}. 102 * 103 * @param time The date in milliseconds elapsed since January 1st, 1970. 104 */ 105 static double julianDay(final long time) { 106 return (time/MILLIS_IN_DAY) + JULIAN_DAY_1970; 107 } 108 109 /** 110 * Retourne le nombre de siècles écoulés depuis le 1 janvier 2000 à midi. 111 * Cette information est utilisée dans les formules de Laskar (1986) pour 112 * calculer la longueur d'une année tropicale, ainsi que par Chapront-Touze 113 * et Chapront (1988) pour la longueur d'un mois synodique. 114 */ 115 static double julianCentury(final Date time) { 116 return ((time.getTime() / MILLIS_IN_DAY) + (JULIAN_DAY_1970 - 2451545.0)) / 36525; 117 } 118 119 /** 120 * Retourne la longueur de l'année tropicale. L'année tropicale est définie comme l'intervalle 121 * moyen entre deux équinoxes vernales (autour du 21 mars dans l'hémisphère nord). Il correspond 122 * au cycle des saisons. Cet intervalle de temps est une <strong>moyenne</strong>. Un cycle réel 123 * peut s'écarter de plusieurs minutes de cette moyenne. Notez aussi qu'une année tropicale 124 * n'est pas identique à une année sidérale, qui est le temps requis par la Terre pour compléter 125 * un orbite autour du Soleil. En l'an 2000, l'année tropicale avait une longueur d'environ 126 * 365,2422 jours tandis que l'année sidérale avait une longueur de 365,2564 jours. 127 * 128 * @param time A date in the year for which to compute the length. 129 * @return The tropical length of the given year. 130 */ 131 public static double tropicalYearLength(final Date time) { 132 final double T = julianCentury(time); 133 return 365.2421896698 + T*(-0.00000615359 + T*(-7.29E-10 + T*(2.64E-10))); 134 } 135 136 /** 137 * Retourne la longueur du mois synodique. Le mois synodique est l'intervalle de temps moyen 138 * entre deux conjonctions de la lune et du soleil. Il correspond au cycle des phases de la 139 * lune. Cet intervalle de temps est une <strong>moyenne</strong>. Un cycle réel peut s'écarter 140 * de plusieurs heures de cette moyenne. 141 * 142 * @param time A date in the month for which to compute the length. 143 * @return The synodic length of the given month. 144 */ 145 public static double synodicMonthLength(final Date time) { 146 final double T=julianCentury(time); 147 return 29.5305888531 + T*(0.00000021621 + T*(-3.64E-10)); 148 } 149 150 /** 151 * Returns a date format to be used by the command line tools. 152 */ 153 static DateFormat getDateFormat() { 154 final DateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss", Locale.CANADA); 155 format.setTimeZone(TimeZone.getTimeZone("UTC")); 156 return format; 157 } 158 159 /** 160 * Prints the length of a tropical year and a synodic month for the given date. 161 * Cette application peut être lancée avec la syntaxe suivante: 162 * This application can be launch from the command line as below: 163 * 164 * {@preformat shell 165 * java org.geotoolkit.nature.Calendar [date] 166 * } 167 * 168 * where <var>date</var> is an optional argument specifying the date and time in the 169 * {@code "yyyy-MM-dd HH:mm:ss"} format, UTC time zone. If this argument is omitted, 170 * then the current time is used. 171 * 172 * @param args The command line argument. 173 * @throws ParseException If the date is not properly formatted. 174 */ 175 public static void main(final String[] args) throws ParseException { 176 final DateFormat format = getDateFormat(); 177 final Date time = (args.length != 0) ? format.parse(args[0]) : new Date(); 178 final PrintStream out = System.out; 179 out.print("Date (UTC) : "); out.println(format.format(time)); 180 out.print("Tropical year: "); out.println(tropicalYearLength(time)); 181 out.print("Synodic month: "); out.println(synodicMonthLength(time)); 182 } 183}