Java时间相关内容学习(2) 自己封装的Calendar接口

本章主要是收藏一些常用的类和接口,包括:万年历(农历、阳历节日、阴历节日)、自定义的Calendar接口。

万年历

源码如下(ChineseCalendar.java):

package com.via.mce.monthcalendar.utils;

import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.HashMap;

/**
 * 农历日历。<br>
 * 将农历从1901年到2100年之间各年、月的大小以及历年节气保存,然后基于这些数据进行计算。<br>
 * <br>
 * 新增了几个用于农历的常量属性字段,可以使用get()方法获取日历对应的值;<br>
 * 农历年、月、日还可以使用set()/add()/roll()方法设置,其他农历属性自动计算;<br>
 * 另外,还提供了getChinese(int field)方法用于获得农历的中文文字(仅适用于农历属性和星期)。<br>
 * <ul>
 * <li>CHINESE_YEAR - 农历年</li>
 * <li>CHINESE_MONTH - 农历月</li>
 * <li>CHINESE_DATE - 农历日</li>
 * <li>CHINESE_SECTIONAL_TERM - 当月的节气</li>
 * <li>CHINESE_PRINCIPLE_TERM - 当月的中气</li>
 * <li>CHINESE_HEAVENLY_STEM - 农历年的天干</li>
 * <li>CHINESE_EARTHLY_BRANCH - 农历年的地支</li>
 * <li>CHINESE_ZODIAC - 农历年的属相</li>
 * <li>CHINESE_TERM_OR_DATE - 如果当天存在一个节气则指示节气,否则如果当天是初一则指示农历月,否则指示农历日</li>
 * </ul>
 * 注意:<br>
 * 由于Calendar类的设定,公历月份从0起始。所有方法都遵循了这一约定。<br>
 * 但所有的农历属性从1起始。即使是在Calendar提供的方法中,农历月也是从1起始的,并以负数表示闰月。<br>
 * clear()方法在某些情况下会导致农历和公历日期不对应或是不能达到预期的重置效果,应尽量避免使用。<br>
 * 使用getSimpleDateString()获得公历日期字符串时,公历月已经修正;<br>
 * 使用getSimpleChineseDateString()获得农历日期字符串时,农历闰月以*表示。<br>
 *
 * @version 0.12 2011-9-5 <br>
 *          <blockquote>修复一个当使用农历正月日期初始化日历时陷入死循环的问题。</blockquote>
 * @version 0.11 2009-12-27 <br>
 *          <blockquote>修复了获取中文农历时未计算农历日期的问题;<br>
 *          加入一个字段CHINESE_TERM_OR_DATE用于模仿台历的显示方式:如果当天有节气则指示节气,如果是初一指示农历月,
 *          否则指示农历日。</blockquote>
 * @version 0.10 2009-12-22
 */
public final class ChineseCalendar extends GregorianCalendar {
    private static final long serialVersionUID = 8L;

    /** 农历年 */
    public static final int CHINESE_YEAR = 801;
    /** 农历月 */
    public static final int CHINESE_MONTH = 802;
    /** 农历日 */
    public static final int CHINESE_DATE = 803;
    /** 当月的节气对应的公历日(前一个节气) */
    public static final int CHINESE_SECTIONAL_TERM = 804;
    /** 当月的中气对应的公历日(后一个节气) */
    public static final int CHINESE_PRINCIPLE_TERM = 805;
    /** 天干 */
    public static final int CHINESE_HEAVENLY_STEM = 806;
    /** 地支 */
    public static final int CHINESE_EARTHLY_BRANCH = 807;
    /** 农历年的属相(生肖) */
    public static final int CHINESE_ZODIAC = 808;
    /** 节气或者农历日 */
    public static final int CHINESE_TERM_OR_DATE = 888;

    // add by skywang
    /** 农历节日 */
    public static final int LUNAR_FESTIVAL = 809;
    /** 阳历节日 */
    public static final int SOLAR_FESTIVAL = 810;
    /** 节气 */
    public static final int CHINESE_TERM = 811;
    /** 月或者农历日 */
    public static final int CHINESE_MONTH_OR_DATE = 812;
    /** 节日 或 节气 或 农历日 */
    public static final int FESTIVAL_OR_TERM_OR_DATE = 813;

    private int chineseYear;
    private int chineseMonth; // 1起始,负数表示闰月
    private int chineseDate;
    private int sectionalTerm; // 当月节气的公历日
    private int principleTerm; // 当月中气的公历日

    private boolean areChineseFieldsComputed; // 农历日期是否已经经过计算确认
    private boolean areSolarTermsComputed; // 节气是否已经经过计算确认
    private boolean lastSetChinese; // 最后设置的是不是农历属性
    // 查看本栏目更多精彩内容:http://www.bianceng.cnhttp://www.bianceng.cn/Programming/Java/
    /** 使用当前时间构造一个实例。 */
    public ChineseCalendar() {
        super();
    }

    /** 使用指定时间构造一个实例。 */
    public ChineseCalendar(Date d) {
        super.setTime(d);
    }

    /** 使用指定时间构造一个实例。 */
    public ChineseCalendar(Calendar c) {
        this(c.getTime());
    }

    /** 使用指定公历日期构造一个实例。 */
    public ChineseCalendar(int y, int m, int d) {
        super(y, m, d);
    }

    /**
     * 使用指定日期构造一个实例。
     *
     * @param isChinese
     *            是否为农历日期
     * @param y
     * @param m
     * @param d
     */
    public ChineseCalendar(boolean isChinese, int y, int m, int d) {
        if (isChinese) {
            set(CHINESE_YEAR, y);
            set(CHINESE_MONTH, m);
            set(CHINESE_DATE, d);
        } else {
            set(y, m, d);
        }
    }

    public void set(int field, int value) {
        computeIfNeed(field);

        if (isChineseField(field)) {
            // 农历属性
            switch (field) {
            case CHINESE_YEAR:
                chineseYear = value;
                break;
            case CHINESE_MONTH:
                chineseMonth = value;
                break;
            case CHINESE_DATE:
                chineseDate = value;
                break;
            default:
                throw new IllegalArgumentException("不支持的field设置:" + field);
            }
            lastSetChinese = true;
        } else {
            // 非农历属性
            super.set(field, value);
            lastSetChinese = false;
        }
        areFieldsSet = false;
        areChineseFieldsComputed = false;
        areSolarTermsComputed = false;
    }

    public int get(int field) {
        computeIfNeed(field);

        if (!isChineseField(field)) {
            return super.get(field);
        }

        switch (field) {
        case CHINESE_YEAR:
            return chineseYear;
        case CHINESE_MONTH:
            return chineseMonth;
        case CHINESE_DATE:
            return chineseDate;
        case CHINESE_SECTIONAL_TERM:
            return sectionalTerm;
        case CHINESE_PRINCIPLE_TERM:
            return principleTerm;
        case CHINESE_HEAVENLY_STEM:
            return (chineseYear - 4) % 10 + 1;
        case CHINESE_EARTHLY_BRANCH:
        case CHINESE_ZODIAC:
            return (chineseYear - 4) % 12 + 1;
        case CHINESE_MONTH_OR_DATE:
            if (get(CHINESE_DATE) == 1) {
                return CHINESE_MONTH;
            } else {
                return CHINESE_DATE;
            }
        case CHINESE_TERM_OR_DATE:
            int option;
            if (get(Calendar.DATE) == get(CHINESE_SECTIONAL_TERM)) {
                option = CHINESE_SECTIONAL_TERM;
            } else if (get(Calendar.DATE) == get(CHINESE_PRINCIPLE_TERM)) {
                option = CHINESE_PRINCIPLE_TERM;
            } else if (get(CHINESE_DATE) == 1) {
                option = CHINESE_MONTH;
            } else {
                option = CHINESE_DATE;
            }
            return option;
        default:
            throw new IllegalArgumentException("不支持的field获取:" + field);
        }
    }

    public void add(int field, int amount) {
        computeIfNeed(field);

        if (!isChineseField(field)) {
            super.add(field, amount);
            lastSetChinese = false;
            areChineseFieldsComputed = false;
            areSolarTermsComputed = false;
            return;
        }

        switch (field) {
        case CHINESE_YEAR:
            chineseYear += amount;
            break;
        case CHINESE_MONTH:
            for (int i = 0; i < amount; i++) {
                chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
                if (chineseMonth == 1) {
                    chineseYear++;
                }
            }
            break;
        case CHINESE_DATE:
            int maxDate = daysInChineseMonth(chineseYear, chineseMonth);
            for (int i = 0; i < amount; i++) {
                chineseDate++;
                if (chineseDate > maxDate) {
                    chineseDate = 1;
                    chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
                    if (chineseMonth == 1) {
                        chineseYear++;
                    }
                    maxDate = daysInChineseMonth(chineseYear, chineseMonth);
                }
            }
        default:
            throw new IllegalArgumentException("不支持的field:" + field);
        }

        lastSetChinese = true;
        areFieldsSet = false;
        areChineseFieldsComputed = false;
        areSolarTermsComputed = false;
    }

    public void roll(int field, int amount) {
        computeIfNeed(field);

        if (!isChineseField(field)) {
            super.roll(field, amount);
            lastSetChinese = false;
            areChineseFieldsComputed = false;
            areSolarTermsComputed = false;
            return;
        }

        switch (field) {
        case CHINESE_YEAR:
            chineseYear += amount;
            break;
        case CHINESE_MONTH:
            for (int i = 0; i < amount; i++) {
                chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
            }
            break;
        case CHINESE_DATE:
            int maxDate = daysInChineseMonth(chineseYear, chineseMonth);
            for (int i = 0; i < amount; i++) {
                chineseDate++;
                if (chineseDate > maxDate) {
                    chineseDate = 1;
                }
            }
        default:
            throw new IllegalArgumentException("不支持的field:" + field);
        }

        lastSetChinese = true;
        areFieldsSet = false;
        areChineseFieldsComputed = false;
        areSolarTermsComputed = false;
    }

    /**
     * 获得属性的中文,可以使用的属性字段为DAY_OF_WEEK以及所有农历属性字段。
     *
     * @param field
     * @return
     */
    public String getChinese(int field) {
        computeIfNeed(field);

        switch (field) {
        case CHINESE_YEAR:
            return getChinese(CHINESE_HEAVENLY_STEM)
                    + getChinese(CHINESE_EARTHLY_BRANCH) + "年";
        case CHINESE_MONTH:
            if (chineseMonth > 0)
                return chineseMonthNames[chineseMonth] + "月";
            else
                return "闰" + chineseMonthNames[-chineseMonth] + "月";
        case CHINESE_DATE:
            return chineseDateNames[chineseDate];
        case CHINESE_SECTIONAL_TERM:
            return sectionalTermNames[get(Calendar.MONTH)];
        case CHINESE_PRINCIPLE_TERM:
            return principleTermNames[get(Calendar.MONTH)];
        case CHINESE_HEAVENLY_STEM:
            return stemNames[get(field)];
        case CHINESE_EARTHLY_BRANCH:
            return branchNames[get(field)];
        case CHINESE_ZODIAC:
            return animalNames[get(field)];
        case Calendar.DAY_OF_WEEK:
            return chineseWeekNames[get(field)];
        case CHINESE_TERM_OR_DATE:
            return getChinese(get(CHINESE_TERM_OR_DATE));
        case LUNAR_FESTIVAL:
            return getLunarFestival();
        case SOLAR_FESTIVAL:
            return getSolarFestival();
        case FESTIVAL_OR_TERM_OR_DATE:
            return getFestivalOrTermOrDate();
            // TODO CHECK
        case CHINESE_MONTH_OR_DATE:
            return getChinese(get(CHINESE_MONTH_OR_DATE));
        case CHINESE_TERM:
            return getChineseTerm();
        default:
            throw new IllegalArgumentException("不支持的field中文获取:" + field);
        }
    }

    public String getSimpleGregorianDateString() {
        return new StringBuffer().append(get(YEAR)).append("-")
                .append(get(MONTH) + 1).append("-").append(get(DATE))
                .toString();
    }

    public String getSimpleChineseDateString() {
        return new StringBuffer()
                .append(get(CHINESE_YEAR))
                .append("-")
                .append(get(CHINESE_MONTH) > 0 ? "" + get(CHINESE_MONTH) : "*"
                        + (-get(CHINESE_MONTH))).append("-")
                .append(get(CHINESE_DATE)).toString();
    }

    public String getChineseDateString() {
        return new StringBuffer().append(getChinese(CHINESE_YEAR))
                .append(getChinese(CHINESE_MONTH))
                .append(getChinese(CHINESE_DATE)).toString();
    }

    public String toString() {
        StringBuffer buf = new StringBuffer();
        buf.append(getSimpleGregorianDateString()).append(" | ")
                .append(getChinese(DAY_OF_WEEK)).append(" | [农历]")
                .append(getChineseDateString()).append(" ")
                .append(getChinese(CHINESE_ZODIAC)).append("年 ")
                .append(get(CHINESE_SECTIONAL_TERM)).append("日")
                .append(getChinese(CHINESE_SECTIONAL_TERM)).append(" ")
                .append(get(CHINESE_PRINCIPLE_TERM)).append("日")
                .append(getChinese(CHINESE_PRINCIPLE_TERM));
        return buf.toString();
    }

    /**
     * 判断是不是农历属性
     *
     * @param field
     * @return
     */
    private boolean isChineseField(int field) {
        switch (field) {
        case CHINESE_YEAR:
        case CHINESE_MONTH:
        case CHINESE_DATE:
        case CHINESE_SECTIONAL_TERM:
        case CHINESE_PRINCIPLE_TERM:
        case CHINESE_HEAVENLY_STEM:
        case CHINESE_EARTHLY_BRANCH:
        case CHINESE_ZODIAC:
        case CHINESE_TERM_OR_DATE:
        case CHINESE_MONTH_OR_DATE:
            return true;
        default:
            return false;
        }
    }

    /**
     * 判断是不是与节气有关的属性
     *
     * @param field
     * @return
     */
    private boolean isChineseTermsField(int field) {
        switch (field) {
        case CHINESE_SECTIONAL_TERM:
        case CHINESE_PRINCIPLE_TERM:
        case CHINESE_TERM_OR_DATE:
            return true;
        default:
            return false;
        }
    }

    /**
     * 如果上一次设置的与这次将要设置或获取的属性不是同一类(农历/公历),<br>
     * 例如上一次设置的是农历而现在要设置或获取公历,<br>
     * 则需要先根据之前设置的农历日期计算出公历日期。
     *
     * @param field
     */
    private void computeIfNeed(int field) {
        if (isChineseField(field)) {
            if (!lastSetChinese && !areChineseFieldsComputed) {
                super.complete();
                computeChineseFields();
                areFieldsSet = true;
                areChineseFieldsComputed = true;
                areSolarTermsComputed = false;
            }
            if (isChineseTermsField(field) && !areSolarTermsComputed) {
                computeSolarTerms();
                areSolarTermsComputed = true;
            }
        } else {
            if (lastSetChinese && !areFieldsSet) {
                computeGregorianFields();
                super.complete();
                areFieldsSet = true;
                areChineseFieldsComputed = true;
                areSolarTermsComputed = false;
            }
        }
    }

    /**
     * 使用农历日期计算出公历日期
     */
    private void computeGregorianFields() {
        int y = chineseYear;
        int m = chineseMonth;
        int d = chineseDate;
        areChineseFieldsComputed = true;
        areFieldsSet = true;
        lastSetChinese = false;

        // 调整日期范围
        if (y < 1900)
            y = 1899;
        else if (y > 2100)
            y = 2101;

        if (m < -12)
            m = -12;
        else if (m > 12)
            m = 12;

        if (d < 1)
            d = 1;
        else if (d > 30)
            d = 30;

        int dateint = y * 10000 + Math.abs(m) * 100 + d;
        if (dateint < 19001111) { // 太小
            set(1901, Calendar.JANUARY, 1);
            super.complete();
        } else if (dateint > 21001201) { // 太大
            set(2100, Calendar.DECEMBER, 31);
            super.complete();
        } else {
            if (Math.abs(m) > 12) {
                m = 12;
            }
            int days = ChineseCalendar.daysInChineseMonth(y, m);
            if (days == 0) {
                m = -m;
                days = ChineseCalendar.daysInChineseMonth(y, m);
            }
            if (d > days) {
                d = days;
            }
            set(y, Math.abs(m) - 1, d);
            computeChineseFields();

            int amount = 0;
            while (chineseYear != y || chineseMonth != m) {
                amount += daysInChineseMonth(chineseYear, chineseMonth);
                chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
                if (chineseMonth == 1) {
                    chineseYear++;
                }
            }
            amount += d - chineseDate;

            super.add(Calendar.DATE, amount);
        }
        computeChineseFields();
    }

    /**
     * 使用公历日期计算出农历日期
     */
    private void computeChineseFields() {
        int gregorianYear = internalGet(Calendar.YEAR);
        int gregorianMonth = internalGet(Calendar.MONTH) + 1;
        int gregorianDate = internalGet(Calendar.DATE);

        if (gregorianYear < 1901 || gregorianYear > 2100) {
            return;
        }

        int startYear, startMonth, startDate;
        if (gregorianYear < 2000) {
            startYear = baseYear;
            startMonth = baseMonth;
            startDate = baseDate;
            chineseYear = baseChineseYear;
            chineseMonth = baseChineseMonth;
            chineseDate = baseChineseDate;
        } else {
            // 第二个对应日,用以提高计算效率
            // 公历 2000 年 1 月 1 日,对应农历 4697(1999) 年 11 月 25 日
            startYear = baseYear + 99;
            startMonth = 1;
            startDate = 1;
            chineseYear = baseChineseYear + 99;
            chineseMonth = 11;
            chineseDate = 25;
        }

        int daysDiff = 0;

        // 年
        for (int i = startYear; i < gregorianYear; i++) {
            if (isGregorianLeapYear(i)) {
                daysDiff += 366; // leap year
            } else {
                daysDiff += 365;
            }
        }

        // 月
        for (int i = startMonth; i < gregorianMonth; i++) {
            daysDiff += daysInGregorianMonth(gregorianYear, i - 1);
        }

        // 日
        daysDiff += gregorianDate - startDate;

        chineseDate += daysDiff;

        int lastDate = daysInChineseMonth(chineseYear, chineseMonth);
        while (chineseDate > lastDate) {
            chineseDate -= lastDate;
            chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
            if (chineseMonth == 1) {
                chineseYear++;
            }
            lastDate = daysInChineseMonth(chineseYear, chineseMonth);
        }

    }

    /**
     * 计算节气
     */
    private void computeSolarTerms() {
        int gregorianYear = internalGet(Calendar.YEAR);
        int gregorianMonth = internalGet(Calendar.MONTH);

        if (gregorianYear < 1901 || gregorianYear > 2100) {
            return;
        }
        sectionalTerm = sectionalTerm(gregorianYear, gregorianMonth);
        principleTerm = principleTerm(gregorianYear, gregorianMonth);
    }

    /* 接下来是静态方法~ */
    /**
     * 是否为公历闰年
     *
     * @param year
     * @return
     */
    public static boolean isGregorianLeapYear(int year) {
        boolean isLeap = false;
        if (year % 4 == 0) {
            isLeap = true;
        }
        if (year % 100 == 0) {
            isLeap = false;
        }
        if (year % 400 == 0) {
            isLeap = true;
        }
        return isLeap;
    }

    /**
     * 计算公历年的当月天数,公历月从0起始!
     *
     * @param y
     * @param m
     * @return
     */
    public static int daysInGregorianMonth(int y, int m) {
        int d = daysInGregorianMonth[m];
        if (m == Calendar.FEBRUARY && isGregorianLeapYear(y)) {
            d++; // 公历闰年二月多一天
        }
        return d;
    }

    /**
     * 计算公历年当月的节气,公历月从0起始!
     *
     * @param y
     * @param m
     * @return
     */
    public static int sectionalTerm(int y, int m) {
        m++;
        if (y < 1901 || y > 2100) {
            return 0;
        }
        int index = 0;
        int ry = y - baseYear + 1;
        while (ry >= sectionalTermYear[m - 1][index]) {
            index++;
        }
        int term = sectionalTermMap[m - 1][4 * index + ry % 4];
        if ((ry == 121) && (m == 4)) {
            term = 5;
        }
        if ((ry == 132) && (m == 4)) {
            term = 5;
        }
        if ((ry == 194) && (m == 6)) {
            term = 6;
        }
        return term;
    }

    /**
     * 计算公历年当月的中气,公历月从0起始!
     *
     * @param y
     * @param m
     * @return
     */
    public static int principleTerm(int y, int m) {
        m++;
        if (y < 1901 || y > 2100) {
            return 0;
        }
        int index = 0;
        int ry = y - baseYear + 1;
        while (ry >= principleTermYear[m - 1][index]) {
            index++;
        }
        int term = principleTermMap[m - 1][4 * index + ry % 4];
        if ((ry == 171) && (m == 3)) {
            term = 21;
        }
        if ((ry == 181) && (m == 5)) {
            term = 21;
        }
        return term;
    }

    /**
     * 计算农历年的天数
     *
     * @param y
     * @param m
     * @return
     */
    public static int daysInChineseMonth(int y, int m) {
        // 注意:闰月 m < 0
        int index = y - baseChineseYear + baseIndex;
        int v = 0;
        int l = 0;
        int d = 30;
        if (1 <= m && m <= 8) {
            v = chineseMonths[2 * index];
            l = m - 1;
            if (((v >> l) & 0x01) == 1) {
                d = 29;
            }
        } else if (9 <= m && m <= 12) {
            v = chineseMonths[2 * index + 1];
            l = m - 9;
            if (((v >> l) & 0x01) == 1) {
                d = 29;
            }
        } else {
            v = chineseMonths[2 * index + 1];
            v = (v >> 4) & 0x0F;
            if (v != Math.abs(m)) {
                d = 0;
            } else {
                d = 29;
                for (int i = 0; i < bigLeapMonthYears.length; i++) {
                    if (bigLeapMonthYears[i] == index) {
                        d = 30;
                        break;
                    }
                }
            }
        }
        return d;
    }

    /**
     * 计算农历的下个月
     *
     * @param y
     * @param m
     * @return
     */
    public static int nextChineseMonth(int y, int m) {
        int n = Math.abs(m) + 1;
        if (m > 0) {
            int index = y - baseChineseYear + baseIndex;
            int v = chineseMonths[2 * index + 1];
            v = (v >> 4) & 0x0F;
            if (v == m) {
                n = -m;
            }
        }
        if (n == 13) {
            n = 1;
        }
        return n;
    }

    /* 日历第一天的日期 */
    private static final int baseYear = 1901;
    private static final int baseMonth = 1;
    private static final int baseDate = 1;
    private static final int baseIndex = 0;
    private static final int baseChineseYear = 1900;
    private static final int baseChineseMonth = 11;
    private static final int baseChineseDate = 11;

    /* 中文字符串 */
    private static final String[] chineseWeekNames = { "", "星期日", "星期一", "星期二",
            "星期三", "星期四", "星期五", "星期六" };
    private static final String[] chineseMonthNames = { "", "正", "二", "三", "四",
            "五", "六", "七", "八", "九", "十", "十一", "十二" };
    private static final String[] chineseDateNames = { "", "初一", "初二", "初三",
            "初四", "初五", "初六", "初七", "初八", "初九", "初十", "十一", "十二", "十三", "十四",
            "十五", "十六", "十七", "十八", "十九", "二十", "廿一", "廿二", "廿三", "廿四", "廿五",
            "廿六", "廿七", "廿八", "廿九", "三十" };
    private static final String[] principleTermNames = { "大寒", "雨水", "春分",
            "谷雨", "夏满", "夏至", "大暑", "处暑", "秋分", "霜降", "小雪", "冬至" };
    private static final String[] sectionalTermNames = { "小寒", "立春", "惊蛰",
            "清明", "立夏", "芒种", "小暑", "立秋", "白露", "寒露", "立冬", "大雪" };
    private static final String[] stemNames = { "", "甲", "乙", "丙", "丁", "戊",
            "己", "庚", "辛", "壬", "癸" };
    private static final String[] branchNames = { "", "子", "丑", "寅", "卯", "辰",
            "巳", "午", "未", "申", "酉", "戌", "亥" };
    private static final String[] animalNames = { "", "鼠", "牛", "虎", "兔", "龙",
            "蛇", "马", "羊", "猴", "鸡", "狗", "猪" };

    /* 接下来是数据压缩表~ */
    private static final int[] bigLeapMonthYears = { 6, 14, 19, 25, 33, 36, 38,
            41, 44, 52, 55, 79, 117, 136, 147, 150, 155, 158, 185, 193 };
    private static final char[][] sectionalTermMap = {
            { 7, 6, 6, 6, 6, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 5, 5,
                    5, 5, 5, 4, 5, 5 },
            { 5, 4, 5, 5, 5, 4, 4, 5, 5, 4, 4, 4, 4, 4, 4, 4, 4, 3, 4, 4, 4, 3,
                    3, 4, 4, 3, 3, 3 },
            { 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 6, 5, 5,
                    5, 5, 4, 5, 5, 5, 5 },
            { 5, 5, 6, 6, 5, 5, 5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 4, 4, 5, 5, 4, 4,
                    4, 5, 4, 4, 4, 4, 5 },
            { 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 6, 5, 5,
                    5, 5, 4, 5, 5, 5, 5 },
            { 6, 6, 7, 7, 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5,
                    5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 5 },
            { 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6,
                    7, 7, 6, 6, 6, 7, 7 },
            { 8, 8, 8, 9, 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7,
                    7, 7, 6, 7, 7, 7, 6, 6, 7, 7, 7 },
            { 8, 8, 8, 9, 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7,
                    7, 7, 6, 7, 7, 7, 7 },
            { 9, 9, 9, 9, 8, 9, 9, 9, 8, 8, 9, 9, 8, 8, 8, 9, 8, 8, 8, 8, 7, 8,
                    8, 8, 7, 7, 8, 8, 8 },
            { 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7,
                    7, 7, 6, 6, 7, 7, 7 },
            { 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6,
                    7, 7, 6, 6, 6, 7, 7 } };
    private static final char[][] sectionalTermYear = {
            { 13, 49, 85, 117, 149, 185, 201, 250, 250 },
            { 13, 45, 81, 117, 149, 185, 201, 250, 250 },
            { 13, 48, 84, 112, 148, 184, 200, 201, 250 },
            { 13, 45, 76, 108, 140, 172, 200, 201, 250 },
            { 13, 44, 72, 104, 132, 168, 200, 201, 250 },
            { 5, 33, 68, 96, 124, 152, 188, 200, 201 },
            { 29, 57, 85, 120, 148, 176, 200, 201, 250 },
            { 13, 48, 76, 104, 132, 168, 196, 200, 201 },
            { 25, 60, 88, 120, 148, 184, 200, 201, 250 },
            { 16, 44, 76, 108, 144, 172, 200, 201, 250 },
            { 28, 60, 92, 124, 160, 192, 200, 201, 250 },
            { 17, 53, 85, 124, 156, 188, 200, 201, 250 } };
    private static final char[][] principleTermMap = {
            { 21, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20,
                    20, 20, 20, 20, 20, 19, 20, 20, 20, 19, 19, 20 },
            { 20, 19, 19, 20, 20, 19, 19, 19, 19, 19, 19, 19, 19, 18, 19, 19,
                    19, 18, 18, 19, 19, 18, 18, 18, 18, 18, 18, 18 },
            { 21, 21, 21, 22, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21,
                    20, 20, 20, 21, 20, 20, 20, 20, 19, 20, 20, 20, 20 },
            { 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 21, 20, 20, 20, 20,
                    19, 20, 20, 20, 19, 19, 20, 20, 19, 19, 19, 20, 20 },
            { 21, 22, 22, 22, 21, 21, 22, 22, 21, 21, 21, 22, 21, 21, 21, 21,
                    20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 21, 21 },
            { 22, 22, 22, 22, 21, 22, 22, 22, 21, 21, 22, 22, 21, 21, 21, 22,
                    21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 21 },
            { 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23, 22, 23, 23, 23,
                    22, 22, 23, 23, 22, 22, 22, 23, 22, 22, 22, 22, 23 },
            { 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23,
                    22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 23 },
            { 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23,
                    22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 23 },
            { 24, 24, 24, 24, 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24,
                    23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 23 },
            { 23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23,
                    22, 22, 22, 22, 21, 22, 22, 22, 21, 21, 22, 22, 22 },
            { 22, 22, 23, 23, 22, 22, 22, 23, 22, 22, 22, 22, 21, 22, 22, 22,
                    21, 21, 22, 22, 21, 21, 21, 22, 21, 21, 21, 21, 22 } };
    private static final char[][] principleTermYear = {
            { 13, 45, 81, 113, 149, 185, 201 },
            { 21, 57, 93, 125, 161, 193, 201 },
            { 21, 56, 88, 120, 152, 188, 200, 201 },
            { 21, 49, 81, 116, 144, 176, 200, 201 },
            { 17, 49, 77, 112, 140, 168, 200, 201 },
            { 28, 60, 88, 116, 148, 180, 200, 201 },
            { 25, 53, 84, 112, 144, 172, 200, 201 },
            { 29, 57, 89, 120, 148, 180, 200, 201 },
            { 17, 45, 73, 108, 140, 168, 200, 201 },
            { 28, 60, 92, 124, 160, 192, 200, 201 },
            { 16, 44, 80, 112, 148, 180, 200, 201 },
            { 17, 53, 88, 120, 156, 188, 200, 201 } };

    private static final char[] daysInGregorianMonth = { 31, 28, 31, 30, 31,
            30, 31, 31, 30, 31, 30, 31 };
    private static final char[] chineseMonths = { 0x00, 0x04, 0xad, 0x08, 0x5a,
            0x01, 0xd5, 0x54, 0xb4, 0x09, 0x64, 0x05, 0x59, 0x45, 0x95, 0x0a,
            0xa6, 0x04, 0x55, 0x24, 0xad, 0x08, 0x5a, 0x62, 0xda, 0x04, 0xb4,
            0x05, 0xb4, 0x55, 0x52, 0x0d, 0x94, 0x0a, 0x4a, 0x2a, 0x56, 0x02,
            0x6d, 0x71, 0x6d, 0x01, 0xda, 0x02, 0xd2, 0x52, 0xa9, 0x05, 0x49,
            0x0d, 0x2a, 0x45, 0x2b, 0x09, 0x56, 0x01, 0xb5, 0x20, 0x6d, 0x01,
            0x59, 0x69, 0xd4, 0x0a, 0xa8, 0x05, 0xa9, 0x56, 0xa5, 0x04, 0x2b,
            0x09, 0x9e, 0x38, 0xb6, 0x08, 0xec, 0x74, 0x6c, 0x05, 0xd4, 0x0a,
            0xe4, 0x6a, 0x52, 0x05, 0x95, 0x0a, 0x5a, 0x42, 0x5b, 0x04, 0xb6,
            0x04, 0xb4, 0x22, 0x6a, 0x05, 0x52, 0x75, 0xc9, 0x0a, 0x52, 0x05,
            0x35, 0x55, 0x4d, 0x0a, 0x5a, 0x02, 0x5d, 0x31, 0xb5, 0x02, 0x6a,
            0x8a, 0x68, 0x05, 0xa9, 0x0a, 0x8a, 0x6a, 0x2a, 0x05, 0x2d, 0x09,
            0xaa, 0x48, 0x5a, 0x01, 0xb5, 0x09, 0xb0, 0x39, 0x64, 0x05, 0x25,
            0x75, 0x95, 0x0a, 0x96, 0x04, 0x4d, 0x54, 0xad, 0x04, 0xda, 0x04,
            0xd4, 0x44, 0xb4, 0x05, 0x54, 0x85, 0x52, 0x0d, 0x92, 0x0a, 0x56,
            0x6a, 0x56, 0x02, 0x6d, 0x02, 0x6a, 0x41, 0xda, 0x02, 0xb2, 0xa1,
            0xa9, 0x05, 0x49, 0x0d, 0x0a, 0x6d, 0x2a, 0x09, 0x56, 0x01, 0xad,
            0x50, 0x6d, 0x01, 0xd9, 0x02, 0xd1, 0x3a, 0xa8, 0x05, 0x29, 0x85,
            0xa5, 0x0c, 0x2a, 0x09, 0x96, 0x54, 0xb6, 0x08, 0x6c, 0x09, 0x64,
            0x45, 0xd4, 0x0a, 0xa4, 0x05, 0x51, 0x25, 0x95, 0x0a, 0x2a, 0x72,
            0x5b, 0x04, 0xb6, 0x04, 0xac, 0x52, 0x6a, 0x05, 0xd2, 0x0a, 0xa2,
            0x4a, 0x4a, 0x05, 0x55, 0x94, 0x2d, 0x0a, 0x5a, 0x02, 0x75, 0x61,
            0xb5, 0x02, 0x6a, 0x03, 0x61, 0x45, 0xa9, 0x0a, 0x4a, 0x05, 0x25,
            0x25, 0x2d, 0x09, 0x9a, 0x68, 0xda, 0x08, 0xb4, 0x09, 0xa8, 0x59,
            0x54, 0x03, 0xa5, 0x0a, 0x91, 0x3a, 0x96, 0x04, 0xad, 0xb0, 0xad,
            0x04, 0xda, 0x04, 0xf4, 0x62, 0xb4, 0x05, 0x54, 0x0b, 0x44, 0x5d,
            0x52, 0x0a, 0x95, 0x04, 0x55, 0x22, 0x6d, 0x02, 0x5a, 0x71, 0xda,
            0x02, 0xaa, 0x05, 0xb2, 0x55, 0x49, 0x0b, 0x4a, 0x0a, 0x2d, 0x39,
            0x36, 0x01, 0x6d, 0x80, 0x6d, 0x01, 0xd9, 0x02, 0xe9, 0x6a, 0xa8,
            0x05, 0x29, 0x0b, 0x9a, 0x4c, 0xaa, 0x08, 0xb6, 0x08, 0xb4, 0x38,
            0x6c, 0x09, 0x54, 0x75, 0xd4, 0x0a, 0xa4, 0x05, 0x45, 0x55, 0x95,
            0x0a, 0x9a, 0x04, 0x55, 0x44, 0xb5, 0x04, 0x6a, 0x82, 0x6a, 0x05,
            0xd2, 0x0a, 0x92, 0x6a, 0x4a, 0x05, 0x55, 0x0a, 0x2a, 0x4a, 0x5a,
            0x02, 0xb5, 0x02, 0xb2, 0x31, 0x69, 0x03, 0x31, 0x73, 0xa9, 0x0a,
            0x4a, 0x05, 0x2d, 0x55, 0x2d, 0x09, 0x5a, 0x01, 0xd5, 0x48, 0xb4,
            0x09, 0x68, 0x89, 0x54, 0x0b, 0xa4, 0x0a, 0xa5, 0x6a, 0x95, 0x04,
            0xad, 0x08, 0x6a, 0x44, 0xda, 0x04, 0x74, 0x05, 0xb0, 0x25, 0x54,
            0x03 };

    private String getChineseTerm() {
      if (get(Calendar.DATE) == get(CHINESE_SECTIONAL_TERM)) {
          return sectionalTermNames[get(Calendar.MONTH)];
      } else if (get(Calendar.DATE) == get(CHINESE_PRINCIPLE_TERM)) {
          return principleTermNames[get(Calendar.MONTH)];
      } else
          return null;
    }
    // add by skywang
    private String getLunarFestival() {
        int day = get(CHINESE_DATE);
        int month = get(CHINESE_MONTH);
        String sToday = day < 10 ?  "0" + day:"" + day;
        String sMonth = month<10 ?  "0" +(month):""+(month);

        return lFestival.get(sMonth+sToday);
    }
    private String getSolarFestival() {
        int day = get(Calendar.DATE);
        int month = get(Calendar.MONTH);
        String sToday = day < 10 ?  "0" + day:"" + day;
        String sMonth = month<10 ?  "0" +(month+1):""+(month+1);

        return sFestival.get(sMonth+sToday);
    }
    private String getFestivalOrTermOrDate() {
        String ret;
        if ((ret = getSolarFestival()) != null)
            return ret;
        if ((ret = getLunarFestival()) != null)
            return ret;
        return getChinese(get(CHINESE_TERM_OR_DATE));
    }

    //公历节日
    private static HashMap<String,String> sFestival =new HashMap<String,String>();
    // 农历介入
    private static HashMap<String,String> lFestival =new HashMap<String,String>();
    static {
        sFestival.put("0101","元旦");
        sFestival.put("0214","情人节");
        sFestival.put("0308","妇女节");
        sFestival.put("0312","植树节");
        sFestival.put("0401","愚人节");
        sFestival.put("0501","劳动节");
        sFestival.put("0504","青年节");
        sFestival.put("0601","儿童节");
        sFestival.put("0701","建党节");
        sFestival.put("0801","建军节");
        sFestival.put("0910","教师节");
        sFestival.put("1001","国庆节");
        sFestival.put("1031","万圣节");
//        sFestival.put("1112","孙中山诞辰");
        sFestival.put("1225","圣诞节");      

        lFestival.put("0101","春节");
//        lFestival.put("0102","大年初二");
//        lFestival.put("0103","大年初三");
        lFestival.put("0115","元宵节");
        lFestival.put("0505","端午节");
        lFestival.put("0707","七夕");
        lFestival.put("0815","中秋节");
        lFestival.put("0909","重阳节");
        lFestival.put("1208","腊八节");
//        lFestival.put("1299","除夕");
    }
}

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索int
, private
, static
, return
, field
, 农历
, v5.6.5
js获取农历日期
,以便于您获取更多的相关知识。

时间: 2024-10-31 08:50:18

Java时间相关内容学习(2) 自己封装的Calendar接口的相关文章

Java时间相关内容学习(1) Calendar

Java 操作日期/时间,往往会涉及到Calendar,Date,DateFormat这些类. 最近决定把这些内容系统的整理一下,这样以后使用的时候,会更得心应手.本章的内容是主要讲解"Java时间框架"以及"类Calendar". 在学习Calendar类时,我们先对它有个整体认识,心中建立一个框架,然后再通过示例学习如何使用它. Java 时间架构图 Java 的Calendar, Date和DateFormat的关系图如下: 说明: (01) millisec

Java时间相关内容学习(3) Date

本章主要介绍Date类,并通过示例学习如何使用它.最后,讲解一下UTC.GMT和时区的关系. Date 介绍 Date 定义 public class Date implements java.io.Serializable, Cloneable, Comparable<Date> {} Date 是表示时间的类. 一个Date对象表示一个特定的瞬间,能精确到毫秒.我们可以通过这个特定的瞬间,来获取到Date对应的"年.月.日.时.分.秒".反之亦然,我们也可以通过设置Da

Java时间相关内容学习(6) Locale介绍

本章介绍Locale. 1 Locale 介绍 Locale 表示地区.每一个Locale对象都代表了一个特定的地理.政治和文化地区. 在操作 Date, Calendar等表示日期/时间的对象时,经常会用到:因为不同的区域,时间表示方式都不同. 下面说说Locale对象的3种常用创建方式. 1 获取默认的Locale 使用方法: Locale locale = Locale.getDefault() 2 直接使用Locale的静态对象 Locale.java中提供了以下静态对象 public

Java时间相关内容学习(5) SimpleDateFormat介绍

介绍SimpleDateFormat. SimpleDateFormat 介绍 SimpleDateFormat 是一个格式化Date 以及 解析日期字符串 的工具.它的最常用途是,能够按照指定的格式来对Date进行格式化,然后我们使用可以格式化Date后得到的字符串. 更严格的说,SimpleDateFormat 是一个以与语言环境有关的方式来格式化和解析日期的具体类.它允许进行格式化(日期 -> 文本).解析(文本 -> 日期)和规范化. SimpleDateFormat的构造函数: //

Java时间相关内容学习(4) DateFormat

本章主要介绍DateFormat. DateFormat 介绍 DateFormat 的作用是 格式化并解析"日期/时间".实际上,它是Date的格式化工具,它能帮助我们格式化Date,进而将Date转换成我们想要的String字符串供我们使用 不过DateFormat的格式化Date的功能有限,没有SimpleDateFormat强大:但DateFormat是SimpleDateFormat的父类.所以,我们先对DateFormat有个整体了解,然后再学习SimpleDateForm

Java时间相关内容学习(7) TimeZone简介

本章介绍TimeZone. TimeZone 简介 TimeZone 表示时区偏移量,也可以计算夏令时. 在操作 Date, Calendar等表示日期/时间的对象时,经常会用到TimeZone:因为不同的时区,时间不同. 下面说说TimeZone对象的 2种常用创建方式. 1 获取默认的TimeZone对象 使用方法: TimeZone tz = TimeZone.getDefault() 2 使用 getTimeZone(String id) 方法获取TimeZone对象 使用方法: //

Java中jqGrid 学习笔记整理——进阶篇(二)_java

相关阅读: Java中jqGrid 学习笔记整理--进阶篇(一) 本篇开始正式与后台(java语言)进行数据交互,使用的平台为 JDK:java 1.8.0_71 myEclisp 2015 Stable 2.0 Apache Tomcat-8.0.30 Mysql 5.7 Navicat for mysql 11.2.5(mysql数据库管理工具) 一.数据库部分 1.创建数据库 使用Navicat for mysql创建数据库(使用其他工具或直接使用命令行暂不介绍) 2. 2.创建表 双击打

Java新手如何学习三大框架

Java新手如何学习Spring.Strurs.Hibernate三大框架? 1. 学习方法 1.1 给我一个可以运行的学习Demo 曾经有过面向百度学习框架的同学,一定会有这种感受,哗啦一搜,各种框架文章呀,真好,真好,趁着学习干劲足足的, 赶紧的照着这些文章做呀. 可是,倒腾了几十分钟,妈蛋~ 怎么跑不出来啊, 怎么报错啊? 这个报错的话,文章里没有说怎么解决呀! 不行不行,换篇文章吧,换了好几个,还是没办法做出效果出来. 慢慢的,心里就凉了,一点点的丧失了学习的兴趣和动力,最后就不了了之了

java泛型的学习笔记[2]—实际使用

继上一文<java泛型的学习笔记[1]-基础知识>之后,本文将介绍泛型的一些应用和应用过程中遇到的问题. 在此之前,我们先给出一张类图:   1)泛型类型的子类型问题 我们首先来看这样一句代码.该行代码正确,因为Cat是Animal的子类型 Animal animal=new Cat();// 但是再看下一句代码: AarrayList<Animal> animals=new ArrayList<Cat>();//编译出错        这句代码编译出错,因为虽然因为C