万年历(农历阴历转换)
package com.java; import java.util.Calendar; public class LunarCalendar { public static void main(String[] args) { Solar s = new Solar(); s.getDate(2020, 9); String t = MutualConversion.solarToLundar(2009, 10, 12); String[] str = t.split("-"); Lunar la = new Lunar(Integer.parseInt(str[0]), Integer .parseInt(str[1]), Integer.parseInt(str[2])); System.out.println("\n\n阳历2009-10-12日对应的农历日期为:"+la.toString(1)+" "+la.toWeek()); String t1 = MutualConversion.lundarToSolar(2019, 1, 1); String[] str1 = t1.split("-"); Solar s1 = new Solar(Integer.parseInt(str1[0]), Integer .parseInt(str1[1]), Integer.parseInt(str1[2])); System.out.println("\n农历2019-1-1日对应的阳历日期为:"+s1.toString()+" "+s1.toWeek()); } } // 自定义日历类,其作用是实现阳历和农历日期相互转换的功能 class MutualConversion { // 阵列storeLunarMonth存储在每月一天的资料,每年从1901年到2100年的农历,农历只能是29或30天, // 每月表达12(或13)的二进制位在一年内,这是30 1中的相应位置的形式天,否则,29天 private static final int[]storeLunarMonth = { 0x4ae0,0xa570,0x5268,0xd260,0xd950,0x6aa8, 0x56a0,0x9ad0,0x4ae8,0x4ae0,//1910 0xa4d8,0xa4d0,0xd250,0xd548,0xb550,0x56a0, 0x96d0,0x95b0,0x49b8,0x49b0,//1920 0xa4b0,0xb258,0x6a50,0x6d40,0xada8, 0x2b60,0x9570,0x4978,0x4970,0x64b0, //1930 0xd4a0,0xea50,0x6d48,0x5ad0,0x2b60, 0x9370,0x92e0,0xc968,0xc950,0xd4a0, //1940 0xda50,0xb550,0x56a0,0xaad8,0x25d0, 0x92d0,0xc958,0xa950,0xb4a8,0x6ca0, //1950 0xb550,0x55a8,0x4da0,0xa5b0,0x52b8, 0x52b0,0xa950,0xe950,0x6aa0,0xad50, //1960 0xab50,0x4b60,0xa570,0xa570, 0x5260,0xe930,0xd950,0x5aa8,0x56a0,0x96d0, //1970 0x4ae8,0x4ad0,0xa4d0,0xd268,0xd250, 0xd528,0xb540,0xb6a0,0x96d0,0x95b0, //1980 0x49b0,0xa4b8,0xa4b0,0xb258,0x6a50, 0x6d40,0xada0,0xab60,0x9370,0x4978, // 1990 0x4970,0x64b0,0x6a50,0xea50,0x6b28, 0x5ac0,0xab60,0x9368,0x92e0,0xc960, //2000 0xd4a8,0xd4a0,0xda50,0x5aa8,0x56a0, 0xaad8,0x25d0,0x92d0,0xc958,0xa950, //2010 0xb4a0,0xb550,0xb550,0x55a8,0x4ba0 ,0xa5b0,0x52b8,0x52b0,0xa930,0x74a8, //2020 0x6aa0,0xad50,0x4da8,0x4b60,0x9570, 0xa4e0,0xd260,0xe930,0xd530,0x5aa0, //2030 0x6b50,0x96d0,0x4ae8,0x4ad0,0xa4d0, 0xd258,0xd250,0xd520,0xdaa0,0xb5a0, //2040 0x56d0,0x4ad8,0x49b0,0xa4b8,0xa4b0, 0xaa50,0xb528,0x6d20,0xada0,0x55b0 //2050 }; // 阵列storeLunarLeapMonth存储的是农历1901年至2050年闰月的信息,0表示该年没有闰月,每个字符元素表示的存储两年。 // 例如0x50该字符存储的是2009和2010年,0x表示八进制,5表示2009年闰5月,0表示2010没有闰月 private static final char[] storeLunarLeapMonth = { 0x00, 0x50, 0x04, 0x00, 0x20, // 1910 0x60, 0x05, 0x00, 0x20, 0x70, // 1920 0x05, 0x00, 0x40, 0x02, 0x06, // 1930 0x00, 0x50, 0x03, 0x07, 0x00, // 1940 0x60, 0x04, 0x00, 0x20, 0x70, // 1950 0x05, 0x00, 0x30, 0x80, 0x06, // 1960 0x00, 0x40, 0x03, 0x07, 0x00, // 1970 0x50, 0x04, 0x08, 0x00, 0x60, // 1980 0x04, 0x0a, 0x00, 0x60, 0x05, // 1990 0x00, 0x30, 0x80, 0x05, 0x00, // 2000 0x40, 0x02, 0x07, 0x00, 0x50, // 2010 0x04, 0x09, 0x00, 0x60, 0x04, // 2020 0x00, 0x20, 0x60, 0x05, 0x00, // 2030 0x30, 0xb0, 0x06, 0x00, 0x50, // 2040 0x02, 0x07, 0x00, 0x50, 0x03 // 2050 }; // 用矩阵存领存储从1901~2050年每一年的阳历和农历的偏移天数 private static final char[] offseOfDays = { 49, 38, 28, 46, 34, 24, 43, 32, 21, 40, // 1910 29, 48, 36, 25, 44, 34, 22, 41, 31, 50, // 1920 38, 27, 46, 35, 23, 43, 32, 22, 40, 29, // 1930 47, 36, 25, 44, 34, 23, 41, 30, 49, 38, // 1940 26, 45, 35, 24, 43, 32, 21, 40, 28, 47, // 1950 36, 26, 44, 33, 23, 42, 30, 48, 38, 27, // 1960 45, 35, 24, 43, 32, 20, 39, 29, 47, 36, // 1970 26, 45, 33, 22, 41, 30, 48, 37, 27, 46, // 1980 35, 24, 43, 32, 50, 39, 28, 47, 36, 26, // 1990 45, 34, 22, 40, 30, 49, 37, 27, 46, 35, // 2000 23, 42, 31, 21, 39, 28, 48, 37, 25, 44, // 2010 33, 23, 41, 31, 50, 39, 28, 47, 35, 24, // 2020 42, 30, 21, 40, 28, 47, 36, 25, 43, 33, // 2030 22, 41, 30, 49, 37, 26, 44, 33, 23, 42, // 2040 31, 21, 40, 29, 47, 36, 25, 44, 32, 22, // 2050 }; static boolean isLeapYearOfSolar(int year) {// 判断是否有闰年 return ((year % 4 == 0) && (year % 100 != 0) || year % 400 == 0); } // 获取阳历中每个月的天数 static int getSolarMonthOfDays(int year, int month) { if ((month == 1) || (month == 3) || (month == 5) || (month == 7) || (month == 8) || (month == 10) || (month == 12)) return 31; else if ((month == 4) || (month == 6) || (month == 9) || (month == 11)) return 30; else if (month == 2) { if (isLeapYearOfSolar(year)) return 29; else return 28; } else return 0; } // 获取新年的偏移天 static int L_getOffsetOfDays(int year, int month, int day) { int days = 0; for (int i = 1; i < month; i++) { days += getSolarMonthOfDays(year, i); } days += day - 1; return days; } static int L_getLeapMonth(int year) { char month = storeLunarLeapMonth[(year - 1901) / 2]; if (year % 2 == 0) return (month & 0x0f); else return (month & 0xf0) >> 4; } // 获取当前年月的农历月 static int L_getMonthDays(int year, int month) { int leapMonth = L_getLeapMonth(year); if ((month > 12) && (month - 12 != leapMonth) || (month < 0)) { System.out.println("输入的月份有错误"); return -1; } if (month - 12 == leapMonth) { if ((storeLunarMonth[year - 1901] & (0x8000 >> leapMonth)) == 0) return 29; else return 30; } if ((leapMonth > 0) && (month > leapMonth)) month++; if ((storeLunarMonth[year - 1901] & (0x8000 >> (month - 1))) == 0) return 29; else return 30; } // 获取当前年份的农历年 static int getLunarYear(int year) { int iYearDays = 0; int leapMonth = L_getLeapMonth(year); for (int i = 1; i < 13; i++) iYearDays += L_getMonthDays(year, i); if (leapMonth > 0) iYearDays += L_getMonthDays(year, leapMonth + 12); return iYearDays; } static int getOffsetOfDays(int year, int month, int day) { int days = 0; int leapMonth = L_getLeapMonth(year); if ((leapMonth > 0) && (leapMonth == month - 12)) { month = leapMonth; days += L_getMonthDays(year, month); } for (int i = 1; i < month; i++) { days += L_getMonthDays(year, i); if (i == leapMonth) days += L_getMonthDays(year, leapMonth + 12); } days += day - 1; return days; } // 阳历转换成农历 static String solarToLundar(int year, int month, int day) { int L_day, L_month, L_year; int days = L_getOffsetOfDays(year, month, day); int leapMonth = L_getLeapMonth(year); if (days < offseOfDays[year - 1901]) { L_year = year - 1; days = offseOfDays[year - 1901] - days; L_day = days; for (L_month = 12; days > L_getMonthDays(L_year, L_month); L_month--) { L_day = days; days -= L_getMonthDays(L_year, L_month); } if (0 == L_day) L_day = 1; else L_day = L_getMonthDays(L_year, L_month) - days + 1; } else { L_year = year; days -= offseOfDays[year - 1901]; L_day = days + 1; for (L_month = 1; days >= 0; L_month++) { L_day = days + 1; days -= L_getMonthDays(L_year, L_month); if ((leapMonth == L_month) && (days > 0)) { L_day = days; days -= L_getMonthDays(L_year, L_month + 12); if (days <= 0) { L_month += 12 + 1; break; } } } L_month--; } return "" + L_year + "-" + L_month + "-" + L_day; } // 农历转换成阳历 static String lundarToSolar(int year, int month, int day) { int S_year, S_month, S_day; int days = getOffsetOfDays(year, month, day) + offseOfDays[year - 1901]; int iYearDays = isLeapYearOfSolar(year) ? 366 : 365; if (days >= iYearDays) { S_year = year + 1; days -= iYearDays; } else { S_year = year; } S_day = days + 1; for (S_month = 1; days >= 0; S_month++) { S_day = days + 1; days -= getSolarMonthOfDays(S_year, S_month); } S_month--; return "" + S_year + "-" + S_month + "-" + S_day; } } // 自定义星期类 class CustomWeek { int week; private String weeks[] = { "星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六" };// 定义一个String数组,存储一周七天的星期 public CustomWeek() {// 默认星期为星期日 week = 0; } // 0:星期日 1:星期一 2:星期二 3:星期三 4:星期四 5:星期五 6:星期六 public CustomWeek(int w) { if ((w > 6) || (w < 0)) { System.out .println("CustomWeek out of range, I think you want Sunday"); this.week = 0; } else this.week = w; } public String toString() { return weeks[week]; } } // 自定义日期类 class CustomDate { public int year; public int month; public int day; private static int checkYear(int year) {// 检察输入的年份是否在指定的年份范围中 if ((year > 1901) && (year < 2050)) return year; else { System.out.println("输入的年份不在1901~2050之间,默认年份为1991年"); return 1991; } } // 构造方法 public CustomDate(int year, int month, int day) { this.year = checkYear(year); this.month = month; this.day = day; } public CustomDate(int year, int month) { this.year = checkYear(year); this.month = month; this.day = 1; } public CustomDate(int year) { this.year = checkYear(year); this.month = 1; this.day = 1; } public CustomDate() {// 默认初始化的日期为1991-01-01 this.year = 1991; this.month = 1; this.day = 1; } public String toString() { return "" + this.year + (this.month > 9 ? "" + this.month : "0" + this.month)// 月以MM的形式表示 + (this.day > 9 ? "" + this.day : "0" + this.day);// 日以dd的形式表示 } public boolean equals(CustomDate md) { return ((md.day == this.day) && (md.month == this.month) && (md.year == this.year)); } } // 阳历日期类,继承自定义日期 class Solar extends CustomDate { private static int checkMonth(int month) {// 检查月份是否越有效范围 if (month > 12) { System.out.println("输入的月份越界, 默认月份为12月 "); return 12; } else if (month < 1) { System.out.println("输入的月份越界, 默认月份为1月"); return 1; } else return month; } private static int checkDay(int year, int month, int day) {// 检查日期是否越有效范围 int iMonthDays = MutualConversion.getSolarMonthOfDays(year, month); if (day > iMonthDays) { System.out.println("输入的日期越界, 默认日期为 " + iMonthDays + " "); return iMonthDays; } else if (day < 1) { System.out.println("输入的日期越界, 默认日期为1号"); return 1; } else return day; } // SolarDate类的构造方法 public Solar(int year, int month, int day) { super(year); this.month = checkMonth(month); this.day = checkDay(this.year, this.month, day); } public Solar(int year, int month) { super(year); this.month = checkMonth(month); } public Solar(int year) { super(year); } public Solar() { super(); } // 以字符串的形式输出 public String toString() { return "" + this.year + (this.month > 9 ? "-" + this.month : "-0" + this.month) + (this.day > 9 ? "-" + this.day : "-0" + this.day); } // 获取输入的年月日是星期几 public CustomWeek toWeek() { int days = 0; for (int i = 1901; i < year; i++) { if (MutualConversion.isLeapYearOfSolar(i)) days += 366; else days += 365; } days += MutualConversion.L_getOffsetOfDays(year, month, day); return new CustomWeek((days + 2) % 7); } public Lunar dateToLunar() {// 将输入日期转换成农历日期 int year, month, day, iDate; Lunar ld; iDate = Integer.parseInt(MutualConversion.solarToLundar(this.year, this.month, this.day)); year = iDate / 10000; month = iDate % 10000 / 100; day = iDate % 100; ld = new Lunar(year, month, day); return ld; } public void getDate(int year, int month) { Calendar calendar = Calendar.getInstance();// 获取一个Calendar对象 int count = 0;// 定义一个计数变量 calendar.set(Calendar.YEAR, year);// 设置年份 calendar.set(Calendar.MONTH, month - 1);// 设置月份 calendar.set(Calendar.DATE, 0);// 设置日期 int days = MutualConversion.getSolarMonthOfDays(year, month); ; System.out.println(year + " 年 " + month + " 月 份 的 万 年 历 如 下:"); System.out.println("星期日\t\t星期一\t\t星期二\t\t星期三\t\t星期四\t\t星期五\t\t星期六"); while (count < days) { calendar.add(Calendar.DAY_OF_MONTH, 1);// 设置添加日历的周期为1 int day = calendar.getTime().getDay();// 获取日历的星期几表示数,例如:0:表示星期日 if (count == 0) {// 根据星期几来决定输入几个tab for (int i = 0; i < day; i++) { System.out.print("\t\t"); } } if (day == 0) {// 如果是周日了则换行 System.out.println(); } String time = MutualConversion.solarToLundar(year, month, calendar .getTime().getDate()); String[] str = time.split("-"); Lunar la = new Lunar(Integer.parseInt(str[0]), Integer .parseInt(str[1]), Integer.parseInt(str[2])); String name = la.toString(0); System.out.print(calendar.getTime().getDate() + " ");// 获取日历中日期数 System.out.print(name + "\t");// 获取日历中日期数 count++; } } } // 农历日期类,继承自定义日期类 class Lunar extends CustomDate { private String upperFigure[] = { "零", "一", "二", "三", "四", "五", "六", "七", "八", "九", "十" }; private static int checkDay(int year, int month, int day) {// 检查日期是否越有效范围 int iMonthDays = MutualConversion.getSolarMonthOfDays(year, month); if (day > iMonthDays) { System.out.println("输入的日期越界, 默认日期为 " + iMonthDays + " "); return iMonthDays; } else if (day < 1) { System.out.println("输入的日期越界, 默认日期为1号"); return 1; } else return day; } private static int checkMonth(int year, int month) {// 检查月份是否越有效范围 if ((month > 12) && (month == MutualConversion.L_getLeapMonth(year) + 12)) { return month; } else if (month > 12) { System.out.println("输入的月份越界, 默认月份为12月 "); return 12; } else if (month < 1) { System.out.println("输入的月份越界, 默认月份为1月"); return 1; } else return month; } // LunarDate类的构造方法 public Lunar(int year, int month, int day) { super(year); this.month = checkMonth(this.year, month); this.day = checkDay(this.year, this.month, day); } public Lunar(int year, int month) { super(year); this.month = checkMonth(this.year, month); } public Lunar(int year) { super(year); } public Lunar() { super(); } // 以字符串的形式输出 public String toString(int n) { String sCalendar = ""; if (n == 1) { sCalendar += "农历" + upperFigure[year / 1000] + upperFigure[year % 1000 / 100] + upperFigure[year % 100 / 10] + upperFigure[year % 10] + "(" + toChineseEra() + ")年"; } if (month > 12) { month -= 12; sCalendar += "闰"; } if (month == 12) sCalendar += "腊月"; else if (month == 11) sCalendar += "冬月"; else if (month == 1) sCalendar += "正月"; else sCalendar += upperFigure[month] + "月"; if (day > 29) sCalendar += "三十"; else if (day > 20) sCalendar += "二十" + upperFigure[day % 20]; else if (day == 20) sCalendar += "二十"; else if (day > 10) sCalendar += "十" + upperFigure[day % 10]; else sCalendar += "初" + upperFigure[day]; return sCalendar; } public CnWeek toWeek() {// 获取输入日期的星期几 int days = 0; for (int i = 1901; i < year; i++) days += MutualConversion.getLunarYear(i); days += MutualConversion.getOffsetOfDays(year, month, day); return new CnWeek((days + 2) % 7); } public ChineseEra toChineseEra() { return new ChineseEra(year); } public Solar toSolarDate() {// 转换成阳历 int year, month, day, iDate; Solar sd; iDate = Integer.parseInt(MutualConversion.lundarToSolar(this.year, this.month, this.day)); year = iDate / 10000; month = iDate % 10000 / 100; day = iDate % 100; sd = new Solar(year, month, day); return sd; } } class CnWeek extends CustomWeek {// Week的子类 private String sCnWeek[] = { "日", "一", "二", "三", "四", "五", "六" }; // 调用父类的构造方法 public CnWeek() { super(); } public CnWeek(int week) { super(week); } public String toString() { return "星期" + sCnWeek[this.week]; } } // 用天干地支形式表示农历年 class ChineseEra { int year; String[] westernNotation = { "甲", "乙", "丙", "丁", "戊", "己", "庚", "辛", "壬", "癸" }; String[] chineseEraNotation = { "子", "丑", "寅", "卯", "辰", "巳", "午", "未", "申", "酉", "戌", "亥" }; public ChineseEra() { int year = 1991; } public ChineseEra(int year) { if ((year < 2050) && (year > 1901)) this.year = year; else this.year = 1991; } public String toString() { int temp; temp = Math.abs(year - 1924); return westernNotation[temp % 10] + chineseEraNotation[temp % 12]; } }