最近在看万年历算法,看的主要内容是杨和荣笔记 - 中国农历二百年算法及年历。
参考资料整理了公历算法,在使用时一般都是已知公历年月日,因而公历算法的关键
在于如何求出星期。
代码如下(已验证):
/*******************************************************************************
* Function Name : Gregorian_IsLeapYear
* Description : 计算给定的年是否是闰年
* Input : int y 指定年
* Output : None
* Return : 0 不是闰年 1 是闰年
*******************************************************************************/
char Gregorian_IsLeapYear(int y)
{
//闰年的规则:逢四闰,逢百不闰,逢四百再闰。
char isLeap = 0;
if (0 == y%4) isLeap = 1;
if (0 == y%100) isLeap = 0;
if (0 == y%400) isLeap = 1;
return isLeap;
}
/*******************************************************************************
* Function Name : Gregorian_DaySum
* Description : 计算从年初至指定日期共计多少天(包括指定日期这天),用于计算星
* 期。
* Input : int y 年
* int m 月
* int d 日
* Output : None
* Return : 返回所求天数
*******************************************************************************/
int Gregorian_DaySum(int y, int m, int d)
{
int mday[13] = {0,31,28,31,30,31,30,31,31,30,31,30,31};
int sum = 0;
int i;
if (Gregorian_IsLeapYear(y))
mday[2] = 29;
for (i=1;i<m;i++)
sum += mday[i];
sum += d;
return sum;
}
/*******************************************************************************
* Function Name : Gregorian_GetWeek
* Description : 计算指定年月日是星期几
* Input : int y 年
* int m 月
* int d 日
* Output : None
* Return : 0~6 对应星期日、星期一...星期六
*******************************************************************************/
int Gregorian_GetWeek(int y, int m, int d)
{
int w = 0;//公元一年一月一日是星期一,所以起始值为星期日
int ly,ry;
//闰年次数
ly = (y-1)/4;
ly = ly - (y-1)/100;
ly = ly + (y-1)/400;
//常年次数
ry = y - 1 - ly;
w = w + ry;//常年星期值加一,365%7=1
w = w + 2*ly;//闰年星期值加二,366%7=2
w = w + Gregorian_DaySum(y,m,d);
w = w%7;
return w;
}
复制代码
[
本帖最后由 threeyears 于 2012-1-5 13:20 编辑 ]