万年历的问题实在搞头晕了

万年历的问题实在搞头晕了,明天就要交东西,我现在都还没有下班,饭也没有吃
我今天一下午都在研究这个问题,老想着自己能解决,但能力实在有限,特来请教大家!

<script type="text/javascript">
var Gan=new Array("甲","乙","丙","丁","戊","己","庚","辛","壬","癸");
var Zhi=new Array("子","丑","寅","卯","辰","巳","午","未","申","酉","戌","亥");
function cyclical(num)
{
return Gan[num%10]+Zhi[num%12];
}
function display()
{
var objDate = new Date();
var y = objDate.getYear();
var m = objDate.getMonth()+1;
var d = objDate.getDate();
cY=cyclical(y-1900+36);
cM = cyclical((y-1900)*12+m+12);
var dayCyclical = Date.UTC(y,m-3,1,0,0,0,0)/86400000+25567+10; //这里的减3是我慢慢试出来的,所以才和今天对应上
cD = cyclical(dayCyclical + d);
document.write(y + "年" + m + "月" + d + "日     ");
document.write(cY + "年" + cM + "月" + cD + "日");
}
</script>
<script language="JavaScript" type="text/javascript">
display();
</script>

这是我从一个万年历里分离出来的,但还有Bug。
比如我的代码“2008年4月10日 戊子年丙辰月辛巳日 ”,实际上是“2008年4月10日 戊子年丙辰月庚辰日 ”
请大家帮着调试试

大家可以对照着这个来验证改对了没有:http://site.baidu.com/list/wannianli.htm


问题点数:50 回复次数:19 显示所有回复显示星级回复显示楼主回复 修改 删除 举报 引用 回复


加为好友
发送私信
在线聊天
Navymk
程序员的肚是杂货铺
等级:
可用分等级:乞丐
总技术分:1291
总技术分排名:15764

发表于:2008-06-11 00:20:131楼 得分:10
农历不同于公历,算法很复杂。公历唯一要考虑的就是闰年问题,而这种问题的难度相比农历的计算而言实在是小儿科。我这话的意思是想让你知道弄清农历的原理是首要的。

下面内容摘自《和荣笔记 - 中国农历二百年算法及年历》。以前我搞过c#的农历,所以收集了一些资料。

==============================================================================================================

中国农历规则

中国农历是阴阳历,同时跟天文月和天文年同步。

农历月的天数是一个变数,有时是 29 天,有时是 30 天。

农历每月的第一天是月亮全黑的日子。

农历年由 24 个节气来确定,节气则由太阳的角度来确定。农历的第一个节气叫雨水,定在太阳的角度为 330 度的日子。其余的 23 个节气分别定在太阳的角度每变化 15 度的日子。下面的表格列出了 24 个节气的名称和定义:

命称 角度 公历日期 周期
立春 315 2月 4日
雨水 330 2月19日 29.8天
惊蛰 345 3月 6日
春分 0 3月21日 30.2天
清明 15 4月 5日
谷雨 30 4月20日 30.7天
立夏 45 5月 6日
夏满 60 5月21日 31.2天
芒种 75 6月 6日
夏至 90 6月22日 31.4天
小暑 105 7月 7日
大暑 120 7月23日 31.4天
立秋 135 8月 8日
处暑 150 8月23日 31.1天
白露 165 9月 8日
秋分 180 9月23日 30.7天
寒露 195 10月 8日
霜降 210 10月24日 30.1天
立冬 225 11月 8日
小雪 240 11月22日 29.7天
大雪 255 12月 7日
冬至 270 12月22日 29.5天
小寒 285 1月 6日
大寒 300 1月20日 29.5天

24 个节气中有 12 个是主节气:雨水,春分,谷雨,夏满,夏至,大暑,处暑,秋分,霜降,小雪,冬至,大寒。

农历年跟天文年相差较大。农历常年有十二个农历月,有 353,354,或者 355 天,比天文年少大约 11 天。为了跟天文年同步,每隔三个农历常年左右,必需设一闰年。闰年有十三个月,添加的这个月叫闰月。

农历十二个月的名称分别为:正月,二月,三月,四月,五月,六月,七月,八月,九月,十月,冬月,腊月。

农历闰年闰月的确定比较难,规则有两条:

一:冬至必须落在农历冬月。如果落不上,腊月之前就要添上一个月,成为闰年。
二:如果是闰年,冬月后边第一个不含主节气的月份定为闰月。闰月使用前一月份的名称。
农历年以 60 年为一周期,每年的名称由 10 个天干的一个字和 12 个地支的一个字排列而成。10 天干为:甲,乙,丙,丁,戊,己,庚,辛,壬,癸。12 地支 为:子,丑,寅,卯,辰,巳,午,未,申,酉,戌,亥。12 地支有 12 动物生肖 与其对应:鼠,牛,虎,兔,龙,蛇,马,羊,猴,鸡,狗,猪。

跟据历史记载,农历年已经经过了 78 个周期。今年,公历 1999 年,是第 79 个周期的第 17 年,也就是农历第 4696 年。

总结起来,中国农历有下例 6 条规则:

一:月全黑规则 - 月全黑的日子是农历月的第一天。农历月周期由此而定。
二:24 节气规则 - 24 节气把天文年按太阳角度分成 24 等份,15 度一节气。这个规则确定了农历和天文年的关系。
三:冬至规则 - 冬至必须落在农历冬月。如果落不上,腊月之前就要添上一个月,成为闰年。
四:闰月规则 - 如果是闰年,冬月后边第一个不含主节气的月份定为闰月。
五:60 年周期 - 农历年以 60 年为一周期。
六:规则一和二的计算必须以中国南京紫金山天文台的观察为准。
==============================================================================================================
我标黑的地方注意一下,这些地方说明农历的制定很大程度上取决于天文观察。
我认为仅仅靠几行代码是不可能完成这种推算确的。

所以,有两条路可以选择:
1、也是最常用的,建立一个农历月份和节气日期的矩阵,直接取数值而不是靠推算。这些数据网上很多,你可以搜索一下。我正在用的一个c#的农历就是搜索来的数据,就到2050年为止。
2、根据天文周期计算精确的日月天象情况,并据此来推算农历。这个我正在考虑,至今未果。

如果明天要交活,自己做肯定来不及了。

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
Navymk
程序员的肚是杂货铺
等级:
可用分等级:乞丐
总技术分:1291
总技术分排名:15764

发表于:2008-06-11 00:24:132楼 得分:3
干支年好算,节气也好说,主要是农历月,日子不定。

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
Kennytian
努力学习
等级:
可用分等级:短工
总技术分:98
总技术分排名:94141

发表于:2008-06-11 06:18:153楼 得分:0
所以啊,我昨天搞了一下午都没有搞好这个天数这个问题,不是多了就是少了。

所以请各位帮忙修改修改!我一大清早就起来关注这个帖子了。

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
nanyang9
奋斗… (网站开发Q群:17322384)
等级:
可用分等级:贫农
总技术分:1954
总技术分排名:11443

发表于:2008-06-11 07:54:534楼 得分:20
引用原文:
曾有一位退休博士用了近三十年找寻阴阳历的公式,问遍两岸各大天文台,得不到答案,直到发现中美天文万年历一书。最後,希望我重新整理的中国阴阳历的天文数字和原始程式,能给有求知欲於阴阳历转换方法的读友一个答案,以免得不到答案而遗憾终身。


下面是原作者的推导的大致转换程序:

JScript code<script language="JavaScript" >
var CalendarData=new Array(20);
var madd=new Array(12);
var TheDate=new Date();
var tgString="甲乙丙丁戊己庚辛壬癸";
var dzString="子丑寅卯辰巳午未申酉戌亥";
var numString="一二三四五六七八九十";
var monString="正二三四五六七八九十冬腊";
var weekString="日一二三四五六";
var sx="鼠牛虎兔龙蛇马羊猴鸡狗猪";
var cYear;
var cMonth;
var cDay;
var cHour;
var cDateString;
var DateString;
var Browser=navigator.appName;
function init()
{
CalendarData[0]=0x41A95;
CalendarData[1]=0xD4A;
CalendarData[2]=0xDA5;
CalendarData[3]=0x20B55;
CalendarData[4]=0x56A;
CalendarData[5]=0x7155B;
CalendarData[6]=0x25D;
CalendarData[7]=0x92D;
CalendarData[8]=0x5192B;
CalendarData[9]=0xA95;
CalendarData[10]=0xB4A;
CalendarData[11]=0x416AA;
CalendarData[12]=0xAD5;
CalendarData[13]=0x90AB5;
CalendarData[14]=0x4BA;
CalendarData[15]=0xA5B;
CalendarData[16]=0x60A57;
CalendarData[17]=0x52B;
CalendarData[18]=0xA93;
CalendarData[19]=0x40E95;
madd[0]=0;
madd[1]=31;
madd[2]=59;
madd[3]=90;
madd[4]=120;
madd[5]=151;
madd[6]=181;
madd[7]=212;
madd[8]=243;
madd[9]=273;
madd[10]=304;
madd[11]=334;
}
function GetBit(m,n)
{
return (m>>n)&1;
}
function e2c()
{
var total,m,n,k;
var isEnd=false;
var tmp=TheDate.getYear();
if (tmp<1900) tmp+=1900;
total=(tmp-2001)*365
+Math.floor((tmp-2001)/4)
+madd[TheDate.getMonth()]
+TheDate.getDate()
-23;
if (TheDate.getYear()%4==0&&TheDate.getMonth()>1)
total++;
for(m=0;;m++)
{
k=(CalendarData[m]<0xfff)?11:12;
for(n=k;n>=0;n--)
{
if(total<=29+GetBit(CalendarData[m],n))
{
isEnd=true;
break;
}
total=total-29-GetBit(CalendarData[m],n);
}
if(isEnd)break;
}
cYear=2001 + m;
cMonth=k-n+1;
cDay=total;
if(k==12)
{
if(cMonth==Math.floor(CalendarData[m]/0x10000)+1)
cMonth=1-cMonth;
if(cMonth>Math.floor(CalendarData[m]/0x10000)+1)
cMonth--;
}
cHour=Math.floor((TheDate.getHours()+3)/2);
}
function GetcDateString()
{ var tmp="";
tmp+=tgString.charAt((cYear-4)%10); //年干
tmp+=dzString.charAt((cYear-4)%12); //年支
tmp+="年(";
tmp+=sx.charAt((cYear-4)%12);
tmp+=")";
if(cMonth<1)
{
tmp+="闰";
tmp+=monString.charAt(-cMonth-1);
}
else
tmp+=monString.charAt(cMonth-1);
tmp+="月";
tmp+=(cDay<11)?"初":((cDay<20)?"十":((cDay<30)?"廿":"卅"));
if(cDay%10!=0||cDay==10)
tmp+=numString.charAt((cDay-1)%10);
if(cHour==13)tmp+="夜";
tmp+=dzString.charAt((cHour-1)%12);
tmp+="时";
cDateString=tmp;
return tmp;
}
function GetDateString()
{
var tmp="";
var t1=TheDate.getYear();
if (t1<1900)t1+=1900;
tmp+=t1
+"-"
+(TheDate.getMonth()+1)+"-"
+TheDate.getDate()+" "
+TheDate.getHours()+":"
+((TheDate.getMinutes()<10)?"0":"")
+TheDate.getMinutes()
+" 星期"+weekString.charAt(TheDate.getDay());
DateString=tmp;
return tmp;
}
init();
e2c();
GetDateString();
GetcDateString();
document.write(DateString,"<br>",cDateString);
</script>




修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
zhou_yuchao
allen
等级:
可用分等级:贫农
总技术分:35
总技术分排名:145301

发表于:2008-06-11 08:12:245楼 得分:5
好人,谢谢分享

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
Kennytian
努力学习
等级:
可用分等级:短工
总技术分:98
总技术分排名:94141

发表于:2008-06-11 08:42:426楼 得分:0
nanyang9兄弟非常好,但是要是能以这种“2008年4月10日 戊子年丙辰月庚辰日”格式显示就好。

这个客户很较真儿,年月日非得是天干地支的形式,唉.....

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
wwjj001
nihao
等级:
可用分等级:乞丐
总技术分:75
总技术分排名:104484

发表于:2008-06-11 09:09:537楼 得分:0
年月你都对上了,明显就是农历日期的事,看差了几天加减一下试试~
我是新手只是提个建议~呵呵

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
Kennytian
努力学习
等级:
可用分等级:短工
总技术分:98
总技术分排名:94141

发表于:2008-06-11 10:52:108楼 得分:0
只是这几天我忽悠过去了,我估计时间一长就又会出错,所以大家有没有一个长久的正确算法?

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
familyX
http://www.doulaicha.com
等级:
可用分等级:中农
总技术分:2366
总技术分排名:9253

发表于:2008-06-11 11:03:339楼 得分:2
http://www.doulaicha.com/lifa.php
http://www.doulaicha.com/lifa.php
你可以参考下这个。

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
kangkai9432

等级:
可用分等级:贫农
总技术分:13
总技术分排名:203531

发表于:2008-06-11 13:20:1910楼 得分:10
我做了一个关于一年节假日的,输入年份 例如:2008 就把一年的节假日全部得到。一共两个类。
其中阴历算法是从hao123的万年历所得,周六周日是从Calendar类所得。
HolidayUtil.java Holiday.java
其中HolidayUtil 的这个方法 public static List getHoliday(String year) 就是返回一年节假日
public class HolidayUtil {

public static void main(String[] args) {

List list = getHoliday("2008");
System.out.println("=="+list.size());
for (int i = 0; i < list.size(); i++) {
Holiday holiday = (Holiday) list.get(i);
System.out.println("==" + holiday.getHOLIDAYDATE() + "==" + holiday.getNAME());
}
lYearDays(2008);
System.out.println(lYearDays(2008));
System.out.println(leapDays(2008));
System.out.println(leapMonth(2008));
System.out.println(monthDays(2008, 1));
Calendar date = Calendar.getInstance();
date.set(2001, 4, 22);
LunarDate ld = lunar(date);
System.out.println("" + ld.day);
System.out.println("" + ld.month);
System.out.println("" + ld.year);
System.out.println("" + ld.isLeap);

Calendar rightNow = Calendar.getInstance();

Calendar tomorrow = Calendar.getInstance();
tomorrow.set(2008, 4, 25);
System.out.println("==" + rightNow.get(Calendar.YEAR));
System.out.println(rightNow.get(Calendar.MONTH));
System.out.println(rightNow.get(Calendar.DATE));
System.out.println(rightNow.get(Calendar.DAY_OF_WEEK));
System.out.println(rightNow.get(Calendar.WEEK_OF_MONTH));

System.out.println("==" + tomorrow.get(Calendar.YEAR));
System.out.println(tomorrow.get(Calendar.MONTH));
System.out.println(tomorrow.get(Calendar.DATE));
System.out.println(tomorrow.get(Calendar.DAY_OF_WEEK));
System.out.println(tomorrow.get(Calendar.WEEK_OF_MONTH));

long aa = sTerm(2008, 12);
System.out.println(aa);

}

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
kangkai9432

等级:
可用分等级:贫农
总技术分:13
总技术分排名:203531

发表于:2008-06-11 13:20:5511楼 得分:0
static long[] lunarInfo = { 0x04bd8, 0x04ae0, 0x0a570, 0x054d5, 0x0d260, 0x0d950, 0x16554, 0x056a0, 0x09ad0, 0x055d2, 0x04ae0, 0x0a5b6, 0x0a4d0,
0x0d250, 0x1d255, 0x0b540, 0x0d6a0, 0x0ada2, 0x095b0, 0x14977, 0x04970, 0x0a4b0, 0x0b4b5, 0x06a50, 0x06d40, 0x1ab54, 0x02b60, 0x09570,
0x052f2, 0x04970, 0x06566, 0x0d4a0, 0x0ea50, 0x06e95, 0x05ad0, 0x02b60, 0x186e3, 0x092e0, 0x1c8d7, 0x0c950, 0x0d4a0, 0x1d8a6, 0x0b550,
0x056a0, 0x1a5b4, 0x025d0, 0x092d0, 0x0d2b2, 0x0a950, 0x0b557, 0x06ca0, 0x0b550, 0x15355, 0x04da0, 0x0a5b0, 0x14573, 0x052b0, 0x0a9a8,
0x0e950, 0x06aa0, 0x0aea6, 0x0ab50, 0x04b60, 0x0aae4, 0x0a570, 0x05260, 0x0f263, 0x0d950, 0x05b57, 0x056a0, 0x096d0, 0x04dd5, 0x04ad0,
0x0a4d0, 0x0d4d4, 0x0d250, 0x0d558, 0x0b540, 0x0b6a0, 0x195a6, 0x095b0, 0x049b0, 0x0a974, 0x0a4b0, 0x0b27a, 0x06a50, 0x06d40, 0x0af46,
0x0ab60, 0x09570, 0x04af5, 0x04970, 0x064b0, 0x074a3, 0x0ea50, 0x06b58, 0x055c0, 0x0ab60, 0x096d5, 0x092e0, 0x0c960, 0x0d954, 0x0d4a0,
0x0da50, 0x07552, 0x056a0, 0x0abb7, 0x025d0, 0x092d0, 0x0cab5, 0x0a950, 0x0b4a0, 0x0baa4, 0x0ad50, 0x055d9, 0x04ba0, 0x0a5b0, 0x15176,
0x052b0, 0x0a930, 0x07954, 0x06aa0, 0x0ad50, 0x05b52, 0x04b60, 0x0a6e6, 0x0a4e0, 0x0d260, 0x0ea65, 0x0d530, 0x05aa0, 0x076a3, 0x096d0,
0x04bd7, 0x04ad0, 0x0a4d0, 0x1d0b6, 0x0d250, 0x0d520, 0x0dd45, 0x0b5a0, 0x056d0, 0x055b2, 0x049b0, 0x0a577, 0x0a4b0, 0x0aa50, 0x1b255,
0x06d20, 0x0ada0, 0x14b63 };

private static long[] solarMonth = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };

private static String[] Gan = { "甲", "乙", "丙", "丁", "戊", "己", "庚", "辛", "壬", "癸" };

private static String[] Zhi = { "子", "丑", "寅", "卯", "辰", "巳", "午", "未", "申", "酉", "戌", "亥" };

private static String[] Animals = { "鼠", "牛", "虎", "兔", "龙", "蛇", "马", "羊", "猴", "鸡", "狗", "猪" };

private static String[] solarTerm = { "小寒", "大寒", "立春", "雨水", "惊蛰", "春分", "清明", "谷雨", "立夏", "小满", "芒种", "夏至", "小暑", "大暑", "立秋", "处暑", "白露", "秋分", "寒露",
"霜降", "立冬", "小雪", "大雪", "冬至" };

private static long[] sTermInfo = { 0, 21208, 42467, 63836, 85337, 107014, 128867, 150921, 173149, 195551, 218072, 240693, 263343, 285989, 308563,
331033, 353350, 375494, 397447, 419210, 440795, 462224, 483532, 504758 };

private static String[] nStr1 = { "日", "一", "二", "三", "四", "五", "六", "七", "八", "九", "十" };

private static String[] nStr2 = { "初", "十", "廿", "卅", "□" };

private static String[] monthName = { "一月", "二月", "三月", "四月", "五月", "六月", "七月", "八月", "九月", "十月", "十一月", "十二月" };

// 国历节日 *表示放假日
private static String[] sFtv = { "0101 元旦节", "0202 世界湿地日", "0210 国际气象节", "0214 情人节", "0301 国际海豹日", "0303 全国爱耳日", "0305 学雷锋纪念日", "0308 妇女节",
"0312 植树节 孙中山逝世纪念日", "0314 国际警察日", "0315 消费者权益日", "0317 中国国医节 国际航海日", "0321 世界森林日 消除种族歧视国际日 世界儿歌日", "0322 世界水日", "0323 世界气象日",
"0324 世界防治结核病日", "0325 全国中小学生安全教育日", "0330 巴勒斯坦国土日", "0401 愚人节 全国爱国卫生运动月(四月) 税收宣传月(四月)", "0407 世界卫生日", "0422 世界地球日", "0423 世界图书和版权日",
"0424 亚非新闻工作者日", "0501 劳动节", "0504 青年节", "0505 碘缺乏病防治日", "0508 世界红十字日", "0512 国际护士节", "0515 国际家庭日", "0517 国际电信日", "0518 国际博物馆日",
"0520 全国学生营养日", "0523 国际牛奶日", "0531 世界无烟日", "0601 国际儿童节", "0605 世界环境保护日", "0606 全国爱眼日", "0617 防治荒漠化和干旱日", "0623 国际奥林匹克日", "0625 全国土地日",
"0626 国际禁毒日", "0701 香港回归纪念日 中共诞辰 世界建筑日", "0702 国际体育记者日", "0707 抗日战争纪念日", "0711 世界人口日", "0730 非洲妇女日", "0801 建军节", "0808 中国男子节(爸爸节)",
"0815 抗日战争胜利纪念", "0908 国际扫盲日 国际新闻工作者日", "0909 毛泽东逝世纪念", "0910 中国教师节", "0914 世界清洁地球日", "0916 国际臭氧层保护日", "0918 九·一八事变纪念日", "0920 国际爱牙日",
"0927 世界旅游日", "0928 孔子诞辰", "1001 国庆节 世界音乐日 国际老人节", "1002 国庆节假日 国际和平与民主自由斗争日", "1003 国庆节假日", "1004 世界动物日", "1006 老人节",
"1008 全国高血压日 世界视觉日", "1009 世界邮政日 万国邮联日", "1010 辛亥革命纪念日 世界精神卫生日", "1013 世界保健日 国际教师节", "1014 世界标准日", "1015 国际盲人节(白手杖节)", "1016 世界粮食日",
"1017 世界消除贫困日", "1022 世界传统医药日", "1024 联合国日", "1031 世界勤俭日", "1107 十月社会主义革命纪念日", "1108 中国记者日", "1109 全国消防安全宣传教育日", "1110 世界青年节",
"1111 国际科学与和平周", "1112 孙中山诞辰纪念日", "1114 世界糖尿病日", "1117 国际大学生节 世界学生节", "1120 彝族年", "1121 彝族年 世界问候日 世界电视日", "1122 彝族年",
"1129 国际声援巴勒斯坦人民国际日", "1201 世界艾滋病日", "1203 世界残疾人日", "1205 国际经济和社会发展志愿人员日", "1208 国际儿童电视日", "1209 世界足球日", "1210 世界人权日", "1212 西安事变纪念日",
"1213 南京大屠杀(1937年)纪念日!", "1220 澳门回归纪念", "1221 国际篮球日", "1224 平安夜", "1225 圣诞节", "1226 毛泽东诞辰纪念" };

// 农历节日 表示放假日
private static String[] lFtv = { "0101 春节", "0102 初二", "0115 元宵节", "0505 端午节", "0707 七夕情人节", "0715 中元节", "0815 中秋节", "0909 重阳节", "1208 腊八节", "1223 小年",
"0100 除夕" };

// 某月的第几个星期几
private static String[] wFtv = {
"0150 世界麻风日", // 一月的最后一个星期日(月倒数第一个星期日)
"0520 国际母亲节", "0530 全国助残日", "0630 父亲节", "0730 被奴役国家周", "0932 国际和平日", "0940 国际聋人节 世界儿童日", "0950 世界海事日", "1011 国际住房日",
"1013 国际减轻自然灾害日(减灾日)", "1144 感恩节" };


修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
kangkai9432

等级:
可用分等级:贫农
总技术分:13
总技术分排名:203531

发表于:2008-06-11 13:21:2612楼 得分:0
// 返回所有节日 List <Holiday>
public static List getHoliday(String year) {
List list = new ArrayList();
long y = Long.valueOf(year);
for (long m = 0; m < 12; m++) {

// 增加 solarTerm 节日
for (int i = 0; i < 2; i++) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(year + "-" + (m + 1) + "-" + sTerm(Long.valueOf(year), 2 * m + i));
holiday.setNAME(solarTerm[(int) 2 * (int) m + (int) i]);
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
//第几个星期
int weekTest1 = 0 , weekTest2 = 0, weekTest3 = 0, weekTest4 = 0, weekTest5 = 0, weekTest6 = 0, weekTest7 = 0, weekTest8 = 0, weekTest9 = 0;
long days = solarDays(y, m);
for (long d = 1; d < days + 1; d++) {

Calendar now = Calendar.getInstance();
now.set((int) y, (int) m, (int) d);
LunarDate ld = lunar(now);
String date = addZero(ld.month) + addZero(ld.day);

// 农历节日
for (int i = 0; i < lFtv.length; i++) {
if (lFtv[i].indexOf(date) >= 0) {
Holiday holiday = new Holiday();
String[] str = lFtv[i].split(" ");
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME(str[1]);
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}

}

// 周六,周日
int day_Of_Week = now.get(Calendar.DAY_OF_WEEK);
if (day_Of_Week == 1) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("周日");
holiday.setHOLIDAYTYPE("W");
list.add(holiday);
}
if (day_Of_Week == 7) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("周六");
holiday.setHOLIDAYTYPE("W");
list.add(holiday);
}

// 某月的第几个星期几
// 世界麻风日
if (m == 0 && day_Of_Week == 1) {
int week_Of_Month = now.get(Calendar.WEEK_OF_MONTH);
if (week_Of_Month == 4) {
Calendar test = Calendar.getInstance();
test.set((int) y, (int) m, (int) d);
test.add(Calendar.DAY_OF_MONTH, 7);
int test_week_Of_Month = test.get(Calendar.WEEK_OF_MONTH);
if (test_week_Of_Month > 4) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + (d + 7));
holiday.setNAME("世界麻风日");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
} else {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("世界麻风日");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}
}

int week_Of_Month = now.get(Calendar.WEEK_OF_MONTH);
// 国际母亲节
if (m == 4 && day_Of_Week == 1) {
weekTest1++;
if(weekTest1==2) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("国际母亲节");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

// 全国助残日
if (m == 4 && day_Of_Week == 1 ) {
weekTest2++;
if(weekTest2==3) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("全国助残日");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

// 父亲节
if (m == 5 && day_Of_Week == 1 ) {
weekTest3++;
if(weekTest3==3) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("父亲节");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

// 被奴役国家周
if (m == 6 && day_Of_Week == 1 ) {
weekTest4++;
if(weekTest4==3) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("被奴役国家周");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

// 国际和平日
if (m == 8 && day_Of_Week == 3 ) {
weekTest5++;
if(weekTest5==4) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("国际和平日");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

// 国际聋人节 世界儿童日
if (m == 8 && day_Of_Week == 1 ) {
weekTest6++;
if(weekTest6==4) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("国际聋人节");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);

Holiday holidayTWO = new Holiday();
holidayTWO.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holidayTWO.setNAME("世界儿童日");
holidayTWO.setHOLIDAYTYPE("H");
list.add(holidayTWO);

}
}

// 世界海事日
if (m == 8 && day_Of_Week == 1) {
if (week_Of_Month == 4) {
Calendar test = Calendar.getInstance();
test.set((int) y, (int) m, (int) d);
test.add(Calendar.DAY_OF_MONTH, 7);
int test_week_Of_Month = test.get(Calendar.WEEK_OF_MONTH);
if (test_week_Of_Month > 4) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + (d + 7));
holiday.setNAME("世界海事日");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
} else {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("世界海事日");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}
}

// 1011 国际住房日
if (m == 9 && day_Of_Week == 2 ) {
weekTest7++;
if(weekTest7==1) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("国际住房日");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

// 1013 国际减轻自然灾害日(减灾日)
if (m == 9 && day_Of_Week == 4 ) {
weekTest8++;
if(weekTest8==1) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("国际减轻自然灾害日(减灾日)");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

// 1144 感恩节
if (m == 10 && day_Of_Week == 5 ) {
weekTest9++;
if(weekTest9==4) {
Holiday holiday = new Holiday();
holiday.setHOLIDAYDATE(y + "-" + (m + 1) + "-" + d);
holiday.setNAME("感恩节");
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
}
}

}

}

// 国历节日
for (int i = 0; i < sFtv.length; i++) {
if (sFtv[i].indexOf(" ") > 0) {
Holiday holiday = new Holiday();
String[] str = sFtv[i].split(" ");
holiday.setHOLIDAYDATE(year + "-" + str[0].substring(0, 2) + "-" + str[0].substring(2, 4));
holiday.setNAME(str[1]);
holiday.setHOLIDAYTYPE("H");
list.add(holiday);
for(int j = 2; j < str.length; j++) {
Holiday holidayTwo = new Holiday();
holidayTwo.setHOLIDAYDATE(year + "-" + str[0].substring(0, 2) + "-" + str[0].substring(2, 4));
holidayTwo.setNAME(str[j]);
holidayTwo.setHOLIDAYTYPE("H");
list.add(holidayTwo);
}
}

}

return list;

}


修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
Navymk
程序员的肚是杂货铺
等级:
可用分等级:乞丐
总技术分:1291
总技术分排名:15764

发表于:2008-06-12 09:03:1013楼 得分:0
引用 4 楼 nanyang9 的回复:
引用原文:
曾有一位退休博士用了近三十年找寻阴阳历的公式,问遍两岸各大天文台,得不到答案,直到发现中美天文万年历一书。最後,希望我重新整理的中国阴阳历的天文数字和原始程式,能给有求知欲於阴阳历转换方法的读友一个答案,以免得不到答案而遗憾终身。

不是我说,这话也太假了....公式贴出来看看,阴阳历根本不可能单靠日期推算出来。

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
Kennytian
努力学习
等级:
可用分等级:短工
总技术分:98
总技术分排名:94141

发表于:2008-06-12 10:04:5714楼 得分:0
还是谢谢大家,我自己慢慢来弄,给分!

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
kangkai9432

等级:
可用分等级:贫农
总技术分:13
总技术分排名:203531

发表于:2008-06-12 14:01:4015楼 得分:0
// 小于10 补0
private static String addZero(long i) {
if (i < 10) {
return "0" + i;
} else {
return String.valueOf(i);
}
}

// 返回农历 y年的总天数
private static long lYearDays(long y) {
long i, sum = 348;
for (i = 0x8000; i > 0x8; i >>= 1) {
if ((lunarInfo[(int) y - 1900] & i) != 0)
sum += 1;
else {
sum += 0;
}
}
return (sum + leapDays(y));
}

// 返回农历 y年闰月的天数
private static long leapDays(long y) {
if (leapMonth(y) != 0) {
if ((lunarInfo[(int) y - 1900] & 0x10000) != 0)
return 30;
else {
return 29;
}
} else
return (0);
}

// 返回农历 y年闰哪个月 1-12 , 没闰返回 0
private static long leapMonth(long y) {
return (lunarInfo[(int) y - 1900] & 0xf);
}

// 返回农历 y年m月的总天数
static long monthDays(long y, long m) {
if ((lunarInfo[(int) y - 1900] & (0x10000 >> m)) != 0)
return 30;
else {
return 29;
}
}

// ====================================== 算出农历, 传入日期控件, 返回农历日期控件
// 该控件属性有 .year .month .day .isLeap
private static LunarDate lunar(Calendar calendar) {

Calendar tomorrow = Calendar.getInstance();
tomorrow.set(1900, 0, 31);

LunarDate lunarDate = new LunarDate();
long i, leap = 0, temp = 0;
long offset = (calendar.getTimeInMillis() - tomorrow.getTimeInMillis()) / 86400000;

for (i = 1900; i < 2050 && offset > 0; i++) {
temp = lYearDays(i);
offset -= temp;
}

if (offset < 0) {
offset += temp;
i--;
}

lunarDate.year = i;

leap = leapMonth(i); // 闰哪个月
lunarDate.isLeap = false;

for (i = 1; i < 13 && offset > 0; i++) {
// 闰月
if (leap > 0 && i == (leap + 1) && lunarDate.isLeap == false) {
--i;
lunarDate.isLeap = true;
temp = leapDays(lunarDate.year);
} else {
temp = monthDays(lunarDate.year, i);
}

// 解除闰月
if (lunarDate.isLeap == true && i == (leap + 1))
lunarDate.isLeap = false;

offset -= temp;
}

if (offset == 0 && leap > 0 && i == leap + 1)
if (lunarDate.isLeap) {
lunarDate.isLeap = false;
} else {
lunarDate.isLeap = true;
--i;
}

if (offset < 0) {
offset += temp;
--i;
}

lunarDate.month = i;
lunarDate.day = (long) offset + 1;
return lunarDate;
}

// ==============================返回公历 y年某m+1月的天数
private static long solarDays(long y, long m) {
if (m == 1)
return (((y % 4 == 0) && (y % 100 != 0) || (y % 400 == 0)) ? 29 : 28);
else
return (solarMonth[(int) m]);
}

// ============================== 传入 offset 返回干支, 0=甲子
private static String cyclical(long num) {
return (Gan[(int) num % 10] + Zhi[(int) num % 12]);
}

// ===== 某年的第n个节气为几日(从0小寒起算)
private static long sTerm(long y, long n) {

Calendar tomorrow = Calendar.getInstance();
Calendar day = Calendar.getInstance();
tomorrow.set(1900, 0, 6, 2, 5);
day.setTimeInMillis((long) (31556925974.7 * (y - 1900) + sTermInfo[(int) n] * 60000) + tomorrow.getTimeInMillis());
return (day.get(Calendar.DAY_OF_MONTH));
}
}

class LunarDate {
long year;

long month;

long day;

boolean isLeap;

}


修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
kangkai9432

等级:
可用分等级:贫农
总技术分:13
总技术分排名:203531

发表于:2008-06-12 14:02:1016楼 得分:0
public class Holiday implements Serializable {
public Holiday() {
}
public String HOLIDAYDATE;
public String HOLIDAYID;
public String HOLIDAYTYPE;
public String COLOR;
public String NAME;
public String STS;
public String STSDATE;


public String getHOLIDAYDATE() {
return HOLIDAYDATE;
}
public void setHOLIDAYDATE(String HOLIDAYDATE) {
this.HOLIDAYDATE = HOLIDAYDATE;
}
public String getHOLIDAYID() {
return HOLIDAYID;
}
public void setHOLIDAYID(String HOLIDAYID) {
this.HOLIDAYID = HOLIDAYID;
}
public String getHOLIDAYTYPE() {
return HOLIDAYTYPE;
}
public void setHOLIDAYTYPE(String HOLIDAYTYPE) {
this.HOLIDAYTYPE = HOLIDAYTYPE;
}

public String getCOLOR() {
return COLOR;
}
public void setCOLOR(String COLOR) {
this.COLOR = COLOR;
}
public String getNAME() {
return NAME;
}
public void setNAME(String NAME) {
this.NAME = NAME;
}
public String getSTS() {
return STS;
}
public void setSTS(String STS) {
this.STS = STS;
}
public String getSTSDATE() {
return STSDATE;
}
public void setSTSDATE(String STSDATE) {
this.STSDATE = STSDATE;
}


}


修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
kangkai9432

等级:
可用分等级:贫农
总技术分:13
总技术分排名:203531

发表于:2008-06-12 14:03:5617楼 得分:0
他妈的 csdn 真气人,多发几个帖子,就发布不上去了,提示说刷帖。 郁闷

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
nanyang9
奋斗… (网站开发Q群:17322384)
等级:
可用分等级:贫农
总技术分:1954
总技术分排名:11443

发表于:2008-06-13 02:00:0218楼 得分:0
引用 13 楼 Navymk 的回复:
引用 4 楼 nanyang9 的回复:
引用原文:
曾有一位退休博士用了近三十年找寻阴阳历的公式,问遍两岸各大天文台,得不到答案,直到发现中美天文万年历一书。最後,希望我重新整理的中国阴阳历的天文数字和原始程式,能给有求知欲於阴阳历转换方法的读友一个答案,以免得不到答案而遗憾终身。

不是我说,这话也太假了....公式贴出来看看,阴阳历根本不可能单靠日期推算出来。


算法在后面是给出来了的.
不过你没理解原作者的意思.
原作者表达的意思也是:不存在绝对的阴阳历转换公式
但原作者不愿看到那位博士及类似的读者失望而归,所以给了一个在一段间隔时间内有效的近似转换程序,而且刚好是JavaScript脚本.

修改 删除 举报 引用 回复

加为好友
发送私信
在线聊天
Navymk
程序员的肚是杂货铺
等级:
可用分等级:乞丐
总技术分:1291
总技术分排名:15764

发表于:2008-06-19 03:20:2119楼 得分:0
引用 18 楼 nanyang9 的回复:
引用 13 楼 Navymk 的回复:
引用 4 楼 nanyang9 的回复:
引用原文:
曾有一位退休博士用了近三十年找寻阴阳历的公式,问遍两岸各大天文台,得不到答案,直到发现中美天文万年历一书。最後,希望我重新整理的中国阴阳历的天文数字和原始程式,能给有求知欲於阴阳历转换方法的读友一个答案,以免得不到答案而遗憾终身。

不是我说,这话也太假了....公式贴出来看看,阴阳历根本不可能单靠日期推算出来。


算法在后面是给出…

权益之计总是够用的了,就这样了,暂时也没办法...话冲一点,没别的意思...

加支付宝好友偷能量挖...


评论(0)网络
阅读(122)喜欢(1)JavaScript/Ajax开发技巧