本文介紹了用查表法實(shí)現(xiàn)的公歷到農(nóng)歷日期轉(zhuǎn)換的方法,給出了實(shí)用的JScript腳本。
可接受的公歷日期范圍是2001-1-1至2050-12-31。
// 數(shù)組LunarDaysOfMonth存入農(nóng)歷2001年到2050年每年中的月天數(shù)信息
// 農(nóng)歷每月只能是29或30天,一年用12(或13)個二進(jìn)制位表示,從高到低,對應(yīng)位為1表示30天,否則29天
var LunarDaysOfMonth = new Array
(
0xd4a8, 0xd4a0, 0xda50, 0x5aa8, 0x56a0, 0xaad8, 0x25d0, 0x92d0, 0xc958, 0xa950, // 2001-2010
0xb4a0, 0xb550, 0xb550, 0x55a8, 0x4ba0, 0xa5b0, 0x52b8, 0x52b0, 0xa930, 0x74a8, // 2011-2020
0x6aa0, 0xad50, 0x4da8, 0x4b60, 0x9570, 0xa4e0, 0xd260, 0xe930, 0xd530, 0x5aa0, // 2021-2030
0x6b50, 0x96d0, 0x4ae8, 0x4ad0, 0xa4d0, 0xd258, 0xd250, 0xd520, 0xdaa0, 0xb5a0, // 2031-2040
0x56d0, 0x4ad8, 0x49b0, 0xa4b8, 0xa4b0, 0xaa50, 0xb528, 0x6d20, 0xada0, 0x55b0 // 2041-2050
);
// 數(shù)組LunarLeapYear存放農(nóng)歷2001年到2050年閏月的月份,如沒有則為0,從高到低,每字節(jié)存兩年
var LunarLeapYear = new Array
(
0x40, 0x02, 0x07, 0x00, 0x50, // 2001-2010
0x04, 0x09, 0x00, 0x60, 0x04, // 2011-2020
0x00, 0x20, 0x60, 0x05, 0x00, // 2021-2030
0x30, 0xb0, 0x06, 0x00, 0x50, // 2031-2040
0x02, 0x07, 0x00, 0x50, 0x03 // 2041-2050
);
// 返回農(nóng)歷iLunarYear年的閏月月份,如沒有則返回0
function GetLeapMonth(iLunarYear)
{
var Leap = LunarLeapYear[(iLunarYear - 2001) >> 1];
return (((iLunarYear - 2001) & 1) == 0) ? (Leap >> 4) : (Leap & 0x0f);
}
// 返回農(nóng)歷iLunarYer年iLunarMonth月的天數(shù),結(jié)果是一個長整數(shù)
// 如果iLunarMonth不是閏月, 高字為0,低字為該月的天數(shù)
// 如果iLunarMonth是閏月, 高字為后一個月的天數(shù),低字為前一個月的天數(shù)
function LunarMonthDays(iLunarYear, iLunarMonth)
{
var High;
var Low;
var Bit;
High = 0;
Low = 29;
Bit = 16 - iLunarMonth;
if ((iLunarMonth > GetLeapMonth(iLunarYear)) && (GetLeapMonth(iLunarYear) > 0)) Bit--;
if ((LunarDaysOfMonth[iLunarYear - 2001] & (1 << Bit)) > 0) Low++;
if (iLunarMonth == GetLeapMonth(iLunarYear))
{
High = ((LunarDaysOfMonth[iLunarYear - 2001] & (1 << (Bit-1))) > 0) ? 30 : 29;
}
return Low + (High << 16);
}
// 返回農(nóng)歷iLunarYear年的總天數(shù)
function LunarYearDays(iLunarYear)
{
var Days;
var tmp;
Days = 0;
for (var i=1; i <= 12; i++)
{
tmp = LunarMonthDays(iLunarYear, i);
Days = Days + ((tmp >> 16) & 0xffff); //取高位
Days = Days + (tmp & 0xffff); //取低位
}
return Days;
}
// 將農(nóng)歷iLunarYear年格式化成天干地支記年法表示的字符串
function FormatLunarYear(iLunarYear)
{
var szText1 = new String("甲乙丙丁戊己庚辛壬癸");
var szText2 = new String("子丑寅卯辰巳午未申酉戌亥");
var strYear;
strYear = szText1.substr((iLunarYear - 4) % 10, 1);
strYear = strYear + szText2.substr((iLunarYear - 4) % 12, 1);
return strYear + "年";
}
// 將農(nóng)歷iLunarMonth月格式化成農(nóng)歷表示的字符串
function FormatLunarMonth(iLunarMonth)
{
var szText = new String("正二三四五六七八九十");
var strMonth;
if (iLunarMonth <= 10)
{
strMonth = szText.substr(iLunarMonth - 1, 1);
}
else if (iLunarMonth == 11) strMonth = "十一";
else strMonth = "十二";
return strMonth + "月";
}
// 將農(nóng)歷iLunarDay日格式化成農(nóng)歷表示的字符串
function FormatLunarDay(iLunarDay)
{
var szText1 = new String("初十廿三");
var szText2 = new String("一二三四五六七八九十");
var strDay;
if ((iLunarDay != 20) && (iLunarDay != 30))
{
strDay = szText1.substr((iLunarDay - 1) / 10, 1) + szText2.substr((iLunarDay - 1) % 10, 1);
}
else if (iLunarDay != 20)
{
strDay = szText1.substr(iLunarDay / 10, 1) + "十";
}
else
{
strDay = "二十";
}
return strDay;
}
// 將公歷日期轉(zhuǎn)換為農(nóng)歷日期,返回農(nóng)歷表示的字符串
function GetLunarDateString(SolarDate)
{
var tmp;
var iLunarYear;
var iLunarMonth;
var iLunarDay;
var Leap = false;
var MinMilli = 1000 * 60;
var HrMilli = MinMilli * 60;
var DyMilli = HrMilli * 24;
// 從2001年1月1日算起,給定的公歷日期已經(jīng)過去的天數(shù)
// 11323是1970年1月1日到2001年1月1日之間的天數(shù),因為Date是從1970年1月1日作為起點(diǎn)的
var iSpanDays = Math.round(SolarDate.getTime() / DyMilli) - 11323;
// 公歷2001年1月24日為農(nóng)歷2001年正月初一,差23天
if (iSpanDays < 23)
{
iYear = 2000;
iLunarMonth = 12;
iLunarDay = iSpanDays + 7;
}
else
{
// 從農(nóng)歷2001年正月初一算起
iSpanDays = iSpanDays - 23;
iLunarYear = 2001;
iLunarMonth = 1;
iLunarDay = 1;
// 計算農(nóng)歷年
tmp = LunarYearDays(iLunarYear);
while (iSpanDays >= tmp)
{
iSpanDays -= tmp;
iLunarYear++;
tmp = LunarYearDays(iLunarYear);
}
// 計算農(nóng)歷月
tmp = LunarMonthDays(iLunarYear, iLunarMonth) & 0xffff; //取低字
while (iSpanDays >= tmp)
{
iSpanDays -= tmp;
if (iLunarMonth == GetLeapMonth(iLunarYear)) // 該年該月閏月
{
tmp = LunarMonthDays(iLunarYear, iLunarMonth) >> 16; //取高字
if (iSpanDays < tmp)
{
Leap = (tmp > 0) ? true : false; // 閏月的后個月?
break;
}
iSpanDays = iSpanDays - tmp;
}
iLunarMonth++;
tmp = LunarMonthDays(iLunarYear,iLunarMonth) & 0xffff; //取低字
}
// 計算農(nóng)歷日
iLunarDay += iSpanDays;
}
return FormatLunarYear(iLunarYear) + (Leap ? "閏" : "") + FormatLunarMonth(iLunarMonth) + FormatLunarDay(iLunarDay);
}
調(diào)用方法舉例如下:
var today= new Date(); // 今天是2004-3-5
var str = GetLunarDateString(today);
結(jié)果是 “甲申年二月十五”。
再舉兩個例子:
var date1 = new Date(2008, 9, 1); // 2008-10-1
var date2 = new Date(2050, 4, 18); // 2050-5-18
var str1 = GetLunarDateString(date1);
var str2 = GetLunarDateString(date2);
結(jié)果分別是 “戊子年九月初三” 和 “庚午年閏三月廿八”。
注意在Date中,月的范圍是0-11。