关闭

Java 计算节气的类

532人阅读 评论(0) 收藏 举报
分类:
  1. /** 
  2.  * @(#)Lunar.java 2008-4-5 2:06:22 
  3.  */  
  4. package luozhuangChineseEra;  
  5.   
  6. import java.text.SimpleDateFormat;  
  7. import java.util.ArrayList;  
  8. import java.util.Arrays;  
  9. import java.util.Calendar;  
  10. import java.util.Date;  
  11. import java.util.GregorianCalendar;  
  12. import java.util.List;  
  13. import java.util.TimeZone;  
  14. import java.util.Vector;  
  15. import java.util.regex.Matcher;  
  16. import java.util.regex.Pattern;  
  17.   
  18. /** 
  19.  * 提供一些农历相关信息 
  20.  * 
  21.  * @author 林志斌,修改by 大师♂罗莊 
  22.  */  
  23. public class Lunar {  
  24.   
  25.     public static void main(String[] args) {  
  26.         Lunar l = new Lunar(System.currentTimeMillis());  
  27.         System.out.println("节气:" + l.getTermString());  
  28.         System.out.println("干支历:" + l.getCyclicalDateString());  
  29.         System.out.println("星期" + l.getDayOfWeek());  
  30.         System.out.println("农历" + l.getLunarDateString());  
  31.         Calendar t = Calendar.getInstance();  
  32.         t.set(19851017);  
  33.         System.out.print(t.getTime().getMonth() + "月");  
  34.         System.out.println(getbeforesolarTerm(t.get(Calendar.YEAR), t.getTime()));  
  35.         System.out.println(getaftersolarTerm(t.get(Calendar.YEAR), t.getTime()));  
  36.   
  37.   
  38.         Date[] jieqi = jieqilist(1940);  
  39.         for (int i = 0; i < solarTerm.length; i++) {  
  40.             System.out.print(solarTerm[i]);  
  41.             System.out.print(jieqi[i].getMonth() + "月");  
  42.             System.out.println(jieqi[i].getDate());  
  43.         }  
  44.     }  
  45.   
  46.     /** 
  47.      * 获得某天前个节气日期差 
  48.      * 
  49.      * @return 日期数 
  50.      */  
  51.     public static long getbeforesolarTerm(int year, Date date) {  
  52.        List<Date> jieqi =Alljieqi(year);  
  53.         int[] jieqibeforeafter = getnearsolarTerm(year, date);  
  54.         return MyDate.DateDays(date, jieqi.get(jieqibeforeafter[0]));  
  55.   
  56.   
  57.     }  
  58.   
  59.     /** 
  60.      * 获得某天后个节气日期差 
  61.      * 
  62.      * @return 日期数 
  63.      */  
  64.     public static long getaftersolarTerm(int year, Date date) {  
  65.         List<Date> jieqi =Alljieqi(year);  
  66.         int[] jieqibeforeafter = getnearsolarTerm(year, date);  
  67.         return MyDate.DateDays(date, jieqi.get(jieqibeforeafter[1]));  
  68.   
  69.     }  
  70.       
  71.       
  72.     public static List<Date> Alljieqi(int year)  
  73.     {  
  74.       
  75.       List<Date> jieqi = new  ArrayList<Date>();  
  76.         Date[] temp ;  
  77.         temp = jieqilist(year - 1);  
  78.         jieqi.addAll(Arrays.asList(temp));  
  79.         temp = jieqilist(year);  
  80.         jieqi.addAll(Arrays.asList(temp));  
  81.          temp = jieqilist(year+1);  
  82.         jieqi.addAll(Arrays.asList(temp));  
  83.         return jieqi;  
  84.     }  
  85.   
  86.     /** 
  87.      * 获得某天前后两个节气序号 
  88.      * 
  89.      * @return 
  90.      */  
  91.     public static int[] getnearsolarTerm(int year, Date date) {  
  92.       List<Date> jieqi =Alljieqi(year);  
  93.           
  94.         int[] returnValue = new int[2];  
  95.         for (int i = 0; i < jieqi.size(); i++) {  
  96.             if (date.getTime() > jieqi.get(i).getTime()) {  
  97.                 continue;  
  98.             }  
  99.             if (i % 2 == 0) {//只管气  
  100.                 returnValue[0] = i - 2;  
  101.                 returnValue[1] = i;  
  102.             } else {  
  103.                 returnValue[0] = i - 1;  
  104.                 returnValue[1] = i + 1;  
  105.   
  106.             }  
  107.             break;  
  108.         }  
  109.   
  110.         return returnValue;  
  111.     }  
  112.   
  113.     /** 
  114.      * 获得某年中所有节气Date 
  115.      * 
  116.      * @return 
  117.      */  
  118.     public static Date[] jieqilist(int year) {  
  119.         Date[] returnvalue = new Date[solarTerm.length];  
  120.   
  121.         for (int i = 0; i < solarTerm.length; i++) {  
  122.   
  123.             Date t = getSolarTermCalendar(year, i);  
  124.             returnvalue[i] = t;  
  125.   
  126.         }  
  127.         return returnvalue;  
  128.     }  
  129.     private final static int[] lunarInfo = {  
  130.         0x4bd80x4ae00xa5700x54d50xd2600xd9500x55540x56af,  
  131.         0x9ad00x55d20x4ae00xa5b60xa4d00xd2500xd2950xb54f,  
  132.         0xd6a00xada20x95b00x49770x497f0xa4b00xb4b50x6a50,  
  133.         0x6d400xab540x2b6f0x95700x52f20x49700x65660xd4a0,  
  134.         0xea500x6a950x5adf0x2b600x86e30x92ef0xc8d70xc95f,  
  135.         0xd4a00xd8a60xb55f0x56a00xa5b40x25df0x92d00xd2b2,  
  136.         0xa9500xb5570x6ca00xb5500x53550x4daf0xa5b00x4573,  
  137.         0x52bf0xa9a80xe9500x6aa00xaea60xab500x4b600xaae4,  
  138.         0xa5700x52600xf2630xd9500x5b570x56a00x96d00x4dd5,  
  139.         0x4ad00xa4d00xd4d40xd2500xd5580xb5400xb6a00x95a6,  
  140.         0x95bf0x49b00xa9740xa4b00xb27a0x6a500x6d400xaf46,  
  141.         0xab600x95700x4af50x49700x64b00x74a30xea500x6b58,  
  142.         0x5ac00xab600x96d50x92e00xc9600xd9540xd4a00xda50,  
  143.         0x75520x56a00xabb70x25d00x92d00xcab50xa9500xb4a0,  
  144.         0xbaa40xad500x55d90x4ba00xa5b00x51760x52bf0xa930,  
  145.         0x79540x6aa00xad500x5b520x4b600xa6e60xa4e00xd260,  
  146.         0xea650xd5300x5aa00x76a30x96d00x4afb0x4ad00xa4d0,  
  147.         0xd0b60xd25f0xd5200xdd450xb5a00x56d00x55b20x49b0,  
  148.         0xa5770xa4b00xaa500xb2550x6d2f0xada00x4b630x937f,  
  149.         0x49f80x49700x64b00x68a60xea5f0x6b200xa6c40xaaef,  
  150.         0x92e00xd2e30xc9600xd5570xd4a00xda500x5d550x56a0,  
  151.         0xa6d00x55d40x52d00xa9b80xa9500xb4a00xb6a60xad50,  
  152.         0x55a00xaba40xa5b00x52b00xb2730x69300x73370x6aa0,  
  153.         0xad500x4b550x4b6f0xa5700x54e40xd2600xe9680xd520,  
  154.         0xdaa00x6aa60x56df0x4ae00xa9d40xa4d00xd1500xf2520xd520  
  155.     };  
  156.     private final static int[] solarTermInfo = {  
  157.         021208424676383685337107014128867150921,  
  158.         173149195551218072240693263343285989308563331033,  
  159.         353350375494397447419210440795462224483532504758  
  160.     };  
  161.     public final static String[] Tianan = {  
  162.         "甲""乙""丙""丁""戊""己""庚""辛""壬""癸"  
  163.     };  
  164.     public final static String[] Deqi = {  
  165.         "子""丑""寅""卯""辰""巳""午""未""申""酉""戌""亥"  
  166.     };  
  167.     public final static String[] Animals = {  
  168.         "鼠""牛""虎""兔""龙""蛇""马""羊""猴""鸡""狗""猪"  
  169.     };  
  170.     public final static String[] solarTerm = {  
  171.         "小寒""大寒""立春""雨水""惊蛰""春分",  
  172.         "清明""谷雨""立夏""小满""芒种""夏至",  
  173.         "小暑""大暑""立秋""处暑""白露""秋分",  
  174.         "寒露""霜降""立冬""小雪""大雪""冬至"  
  175.     };  
  176.     public final static String[] lunarString1 = {  
  177.         "零""一""二""三""四""五""六""七""八""九"  
  178.     };  
  179.     public final static String[] lunarString2 = {  
  180.         "初""十""廿""卅""正""腊""冬""闰"  
  181.     };  
  182.     /** 
  183.      * 国历节日 *表示放假日 
  184.      */  
  185.     private final static String[] sFtv = {  
  186.         "0101*元旦""0214 情人节""0308 妇女节""0312 植树节",  
  187.         "0315 消费者权益日""0401 愚人节""0501*劳动节""0504 青年节",  
  188.         "0509 郝维节""0512 护士节""0601 儿童节""0701 建党节 香港回归纪念",  
  189.         "0801 建军节""0808 父亲节""0816 燕衔泥节""0909 毛泽东逝世纪念",  
  190.         "0910 教师节""0928 孔子诞辰""1001*国庆节""1006 老人节",  
  191.         "1024 联合国日""1111 光棍节""1112 孙中山诞辰纪念""1220 澳门回归纪念",  
  192.         "1225 圣诞节""1226 毛泽东诞辰纪念"  
  193.     };  
  194.     /** 
  195.      * 农历节日 *表示放假日 
  196.      */  
  197.     private final static String[] lFtv = {  
  198.         "0101*春节、弥勒佛诞""0106 定光佛诞""0115 元宵节",  
  199.         "0208 释迦牟尼佛出家""0215 释迦牟尼佛涅槃""0209 海空上师诞",  
  200.         "0219 观世音菩萨诞""0221 普贤菩萨诞""0316 准提菩萨诞",  
  201.         "0404 文殊菩萨诞""0408 释迦牟尼佛诞""0415 佛吉祥日——释迦牟尼佛诞生、成道、涅槃三期同一庆(即南传佛教国家的卫塞节)",  
  202.         "0505 端午节""0513 伽蓝菩萨诞""0603 护法韦驮尊天菩萨诞",  
  203.         "0619 观世音菩萨成道——此日放生、念佛,功德殊胜",  
  204.         "0707 七夕情人节""0713 大势至菩萨诞""0715 中元节",  
  205.         "0724 龙树菩萨诞""0730 地藏菩萨诞""0815 中秋节",  
  206.         "0822 燃灯佛诞""0909 重阳节""0919 观世音菩萨出家纪念日",  
  207.         "0930 药师琉璃光如来诞""1005 达摩祖师诞""1107 阿弥陀佛诞",  
  208.         "1208 释迦如来成道日,腊八节""1224 小年",  
  209.         "1229 华严菩萨诞""0100*除夕"  
  210.     };  
  211.     /** 
  212.      * 某月的第几个星期几 
  213.      */  
  214.     private static String[] wFtv = {  
  215.         "0520 母亲节""0716 合作节""0730 被奴役国家周"  
  216.     };  
  217.   
  218.     private static int toInt(String str) {  
  219.         try {  
  220.             return Integer.parseInt(str);  
  221.         } catch (Exception e) {  
  222.             return -1;  
  223.         }  
  224.     }  
  225.     private final static Pattern sFreg = Pattern.compile("^(\\d{2})(\\d{2})([\\s\\*])(.+)$");  
  226.     private final static Pattern wFreg = Pattern.compile("^(\\d{2})(\\d)(\\d)([\\s\\*])(.+)$");  
  227.   
  228.     private synchronized void findFestival() {  
  229.         int sM = this.getSolarMonth();  
  230.         int sD = this.getSolarDay();  
  231.         int lM = this.getLunarMonth();  
  232.         int lD = this.getLunarDay();  
  233.         int sy = this.getSolarYear();  
  234.         Matcher m;  
  235.         for (int i = 0; i < Lunar.sFtv.length; i++) {  
  236.             m = Lunar.sFreg.matcher(Lunar.sFtv[i]);  
  237.             if (m.find()) {  
  238.                 if (sM == Lunar.toInt(m.group(1)) && sD == Lunar.toInt(m.group(2))) {  
  239.                     this.isSFestival = true;  
  240.                     this.sFestivalName = m.group(4);  
  241.                     if ("*".equals(m.group(3))) {  
  242.                         this.isHoliday = true;  
  243.                     }  
  244.                     break;  
  245.                 }  
  246.             }  
  247.         }  
  248.         for (int i = 0; i < Lunar.lFtv.length; i++) {  
  249.             m = Lunar.sFreg.matcher(Lunar.lFtv[i]);  
  250.             if (m.find()) {  
  251.                 if (lM == Lunar.toInt(m.group(1)) && lD == Lunar.toInt(m.group(2))) {  
  252.                     this.isLFestival = true;  
  253.                     this.lFestivalName = m.group(4);  
  254.                     if ("*".equals(m.group(3))) {  
  255.                         this.isHoliday = true;  
  256.                     }  
  257.                     break;  
  258.                 }  
  259.             }  
  260.         }  
  261.   
  262.         // 月周节日     
  263.         int w, d;  
  264.         for (int i = 0; i < Lunar.wFtv.length; i++) {  
  265.             m = Lunar.wFreg.matcher(Lunar.wFtv[i]);  
  266.             if (m.find()) {  
  267.                 if (this.getSolarMonth() == Lunar.toInt(m.group(1))) {  
  268.                     w = Lunar.toInt(m.group(2));  
  269.                     d = Lunar.toInt(m.group(3));  
  270.                     if (this.solar.get(Calendar.WEEK_OF_MONTH) == w  
  271.                             && this.solar.get(Calendar.DAY_OF_WEEK) == d) {  
  272.                         this.isSFestival = true;  
  273.                         this.sFestivalName += "|" + m.group(5);  
  274.                         if ("*".equals(m.group(4))) {  
  275.                             this.isHoliday = true;  
  276.                         }  
  277.                     }  
  278.                 }  
  279.             }  
  280.         }  
  281.         if (sy > 1874 && sy < 1909) {  
  282.             this.description = "光绪" + (((sy - 1874) == 1) ? "元" : "" + (sy - 1874));  
  283.         }  
  284.         if (sy > 1908 && sy < 1912) {  
  285.             this.description = "宣统" + (((sy - 1908) == 1) ? "元" : String.valueOf(sy - 1908));  
  286.         }  
  287.         if (sy > 1911 && sy < 1950) {  
  288.             this.description = "民国" + (((sy - 1911) == 1) ? "元" : String.valueOf(sy - 1911));  
  289.         }  
  290.         if (sy > 1949) {  
  291.             this.description = "共和国" + (((sy - 1949) == 1) ? "元" : String.valueOf(sy - 1949));  
  292.         }  
  293.         this.description += "年";  
  294.         this.sFestivalName = this.sFestivalName.replaceFirst("^\\|""");  
  295.         this.isFinded = true;  
  296.     }  
  297.     private boolean isFinded = false;  
  298.     private boolean isSFestival = false;  
  299.     private boolean isLFestival = false;  
  300.     private String sFestivalName = "";  
  301.     private String lFestivalName = "";  
  302.     private String description = "";  
  303.     private boolean isHoliday = false;  
  304.   
  305.     /** 
  306.      * 返回农历年闰月月份 
  307.      * 
  308.      * @param lunarYear 指定农历年份(数字) 
  309.      * @return 该农历年闰月的月份(数字,没闰返回0) 
  310.      */  
  311.     private static int getLunarLeapMonth(int lunarYear) {  
  312.         // 数据表中,每个农历年用16bit来表示,     
  313.         // 前12bit分别表示12个月份的大小月,最后4bit表示闰月     
  314.         // 若4bit全为1或全为0,表示没闰, 否则4bit的值为闰月月份     
  315.         int leapMonth = Lunar.lunarInfo[lunarYear - 1900] & 0xf;  
  316.         leapMonth = (leapMonth == 0xf ? 0 : leapMonth);  
  317.         return leapMonth;  
  318.     }  
  319.   
  320.     /** 
  321.      * 返回农历年闰月的天数 
  322.      * 
  323.      * @param lunarYear 指定农历年份(数字) 
  324.      * @return 该农历年闰月的天数(数字) 
  325.      */  
  326.     private static int getLunarLeapDays(int lunarYear) {  
  327.         // 下一年最后4bit为1111,返回30(大月)     
  328.         // 下一年最后4bit不为1111,返回29(小月)     
  329.         // 若该年没有闰月,返回0     
  330.         return Lunar.getLunarLeapMonth(lunarYear) > 0 ? ((Lunar.lunarInfo[lunarYear - 1899] & 0xf) == 0xf ? 30  
  331.                 : 29)  
  332.                 : 0;  
  333.     }  
  334.   
  335.     /** 
  336.      * 返回农历年的总天数 
  337.      * 
  338.      * @param lunarYear 指定农历年份(数字) 
  339.      * @return 该农历年的总天数(数字) 
  340.      */  
  341.     private static int getLunarYearDays(int lunarYear) {  
  342.         // 按小月计算,农历年最少有12 * 29 = 348天     
  343.         int daysInLunarYear = 348;  
  344.         // 数据表中,每个农历年用16bit来表示,     
  345.         // 前12bit分别表示12个月份的大小月,最后4bit表示闰月     
  346.         // 每个大月累加一天     
  347.         for (int i = 0x8000; i > 0x8; i >>= 1) {  
  348.             daysInLunarYear += ((Lunar.lunarInfo[lunarYear - 1900] & i) != 0) ? 1  
  349.                     : 0;  
  350.         }  
  351.         // 加上闰月天数     
  352.         daysInLunarYear += Lunar.getLunarLeapDays(lunarYear);  
  353.   
  354.         return daysInLunarYear;  
  355.     }  
  356.   
  357.     /** 
  358.      * 返回农历年正常月份的总天数 
  359.      * 
  360.      * @param lunarYear 指定农历年份(数字) 
  361.      * @param lunarMonth 指定农历月份(数字) 
  362.      * @return 该农历年闰月的月份(数字,没闰返回0) 
  363.      */  
  364.     private static int getLunarMonthDays(int lunarYear, int lunarMonth) {  
  365.         // 数据表中,每个农历年用16bit来表示,     
  366.         // 前12bit分别表示12个月份的大小月,最后4bit表示闰月     
  367.         int daysInLunarMonth = ((Lunar.lunarInfo[lunarYear - 1900] & (0x10000 >> lunarMonth)) != 0) ? 30  
  368.                 : 29;  
  369.         return daysInLunarMonth;  
  370.     }  
  371.   
  372.     /** 
  373.      * 取 Date 对象中用全球标准时间 (UTC) 表示的日期 
  374.      * 
  375.      * @param date 指定日期 
  376.      * @return UTC 全球标准时间 (UTC) 表示的日期 
  377.      */  
  378.     public static synchronized int getUTCDay(Date date) {  
  379.         Lunar.makeUTCCalendar();  
  380.         synchronized (utcCal) {  
  381.             utcCal.clear();  
  382.             utcCal.setTimeInMillis(date.getTime());  
  383.             return utcCal.get(Calendar.DAY_OF_MONTH);  
  384.         }  
  385.     }  
  386.     private static GregorianCalendar utcCal = null;  
  387.   
  388.     private static synchronized void makeUTCCalendar() {  
  389.         if (Lunar.utcCal == null) {  
  390.             Lunar.utcCal = new GregorianCalendar(TimeZone.getTimeZone("UTC"));  
  391.         }  
  392.     }  
  393.   
  394.     /** 
  395.      * 返回全球标准时间 (UTC) (或 GMT) 的 1970 年 1 月 1 日到所指定日期之间所间隔的毫秒数。 
  396.      * 
  397.      * @param y 指定年份 
  398.      * @param m 指定月份 
  399.      * @param d 指定日期 
  400.      * @param h 指定小时 
  401.      * @param min 指定分钟 
  402.      * @param sec 指定秒数 
  403.      * @return 全球标准时间 (UTC) (或 GMT) 的 1970 年 1 月 1 日到所指定日期之间所间隔的毫秒数 
  404.      */  
  405.     public static synchronized long UTC(int y, int m, int d, int h, int min, int sec) {  
  406.         Lunar.makeUTCCalendar();  
  407.         synchronized (utcCal) {  
  408.             utcCal.clear();  
  409.             utcCal.set(y, m, d, h, min, sec);  
  410.             return utcCal.getTimeInMillis();  
  411.         }  
  412.     }  
  413.   
  414.     /** 
  415.      * 返回公历年节气的日期 
  416.      * 
  417.      * @param solarYear 指定公历年份(数字) 
  418.      * @param index 指定节气序号(数字,0从小寒算起) 
  419.      * @return 日期(数字,所在月份的第几天) 
  420.      */  
  421.     private static int getSolarTermDay(int solarYear, int index) {  
  422.   
  423.         return Lunar.getUTCDay(getSolarTermCalendar(solarYear, index));  
  424.     }  
  425.   
  426.     /** 
  427.      * 返回公历年节气的日期 
  428.      * 
  429.      * @param solarYear 指定公历年份(数字) 
  430.      * @param index 指定节气序号(数字,0从小寒算起) 
  431.      * @return 日期(数字,所在月份的第几天) 
  432.      */  
  433.     public static Date getSolarTermCalendar(int solarYear, int index) {  
  434.         long l = (long31556925974.7 * (solarYear - 1900)  
  435.                 + solarTermInfo[index] * 60000L;  
  436.         l = l + Lunar.UTC(190006250);  
  437.         return new Date(l);  
  438.     }  
  439.     private Calendar solar;  
  440.     private int lunarYear;  
  441.     private int lunarMonth;  
  442.     private int lunarDay;  
  443.     private boolean isLeap;  
  444.     private boolean isLeapYear;  
  445.     private int solarYear;