分享

万年历设计

 共同成长888 2014-05-08

万年历设计

摘要:

随着电子技术的迅速发展,特别是随大规模集成电路出现,给人类生活带来了根本性的改变。尤其是单片机技术的应用产品已经走进了千家万户。电子万年历的出现给人们的生活带来的诸多方便。本设计主要是以单片机的C++语言进行软件设计,增加了程序的可读性和可移植性,为了便于扩展和更改,软件的设计采用模块化结构,使程序设计的逻辑关系更加简洁明了。系统通过纯DOS界面显示调试结果,所以运行操作比较简单。可以显示公农历日期、星期、节气,天干地支。是一个比较简易适合大众的万年历。

1.引言

目前流行的计算机日历程序,比较典型的是Windows各版本中的日历程序以及基础于该程序所开发的各种应用程序中的日历程序。然而,这些程序都千篇一律的局限在一个很短的时间范围内。(Windows各个版本一般都局限在1980年至2099年这一范围内),但是,在很多情况下,一个时间跨度较大的日历程序是很有参考价值的,本程序在这种背景下开始编辑,其中集成了国际通用日历和中国农历,此外还可以显示星期和加载了部分节日,显示本机准确日期等功能。

2.设计思想:

众所周知,地球绕太阳公转,公转一周历时365天5小时48分46秒。现代国际上普遍采用罗马历法,在罗马历法中人为地规定一年365天,也就是我们所说的平年,为了弥补每一年多出的5小时48分46秒,同时又规定4年中有一年是闰年,闰年为366天(平年的2月份为28天,而闰年的2月份为29天),这样4年有365*3+366=1461天,而地球绕太阳公转4周历时1460天23小时15分4秒,这样,每4年又产生了44分56秒的误差,为了减小影响,历法上又规定,每400年中只存在97个闰年,这样400年中共有365*400+97=146097天,而地球绕太阳公转400周历时146096天21小时6分40秒,较好的弥补了这一缺陷,这样几乎3300年才产生一天的误差。所以在历法规定:年份能被4整除的(即年份为4的倍数)都是闰年,不过,年份以“00”结尾的但年份不能被400整除的不是闰年,即公元100、200、300等都不是闰年,公元400、800、1200等是闰年。这一规定适合于公元后的任何年份,但是不适合于公元前的年份,同时现代历法也认为, 不存在公元0年,公元前1年的第二年为公元1年,并不存在公元0年。

实现年月日及时间的查询与修改,功能类似于Windows的时间和日期。同时具备了阴阳历的转换功能,能显示所要查日期的星期和一些纪念日等的功能。

3.算法分析说明

1. 总天数的算法:首先用if语句判断定义年到输入年之间每一年是否为闰年,是闰年,该年的总天数为366,否则,为355。然后判断输入的年是否为定义年,若是,令总天数S=1,否则,用累加法计算出定义年到输入年之间的总天数,再把输入年的一月到要输出的月份之间的天数累加起来,若该月是闰年中的月份并且该月还大于二月,再使总天数加1,否则,不加,既算出从定义年一月一日到输出年的该月一日的总天数。

2. 输出月份第一天为星期几的算法:使总天数除以7取余加2得几既为星期几,若是7,则为星期日。

3. 算出输出月份第一天为星期几的算法:算出输出月份第一天为星期几后,把该日期以前的位置用空格补上,并总该日起一次输出天数直到月底,该月中的天数加上该月一日为星期几的数字再除以7得0换行,即可完整的输出该月的日历。

4.要用到的函数和语句

1.<stdio.h>,<conio.h>,<math.h> /*头文件*/

2.main() /*主函数*/

3.printf(),

4.scanf()

5.textbackground(),textcolor() /*定义背景和字体颜色*/

6.if 语句

7.for 语句

8.printstar() /*调用函数*/

9.int day_year() /*定义函数*/

10.goto /*循环语句*/

5.流程图

SHAPE \* MERGEFORMAT

SHAPE \* MERGEFORMAT

6.设计分析

如何计算某一天是星期几? —— 蔡勒(Zeller)公式

历史上的某一天是星期几?未来的某一天是星期几?关于这个问题,有很多计算公式(两个通用计算公式和一些分段计算公式),其中最著名的是蔡勒(Zeller)公式。 即:w=y+[y/4]+[c/4]-2c+[26(m+1)/10]+d-1

公式中的符号含义如下,w:星期;c:世纪-1;y:年(两位数);m:月(m大于等于3,小于等于14,即在蔡勒公式中,某年的1、2月要看作上一年的13、14月来计算,比如2003年1月1日要看作2002年的13月1日来计算);d:日;[ ]代表取整,即只要整数部分。(C是世纪数减一,y是年份后两位,M是月份,d是日数。1月和2月要按上一年的13月和 14月来算,这时C和y均按上一年取值。)算出来的W除以7,余数是几就是星期几。如果余数是0,则为星期日。以2049年10月1日(100周年国庆)为例,用蔡勒(Zeller)公式进行计算,过程如下:

蔡勒(Zeller)公式:w=y+[y/4]+[c/4]-2c+[26(m+1)/10]+d-1

=49+[49/4]+[20/4]-2×20+[26× (10+1)/10]+1-1

=49+[12.25]+5-40+[28.6]

=49+12+5-40+28

=54 (除以7余5)

即2049年10月1日(100周年国庆)是星期5。

7.功能过程的推导:

星期制度是一种有古老传统的制度。据说因为《圣经·创世纪》中规定上帝用了六 天时间创世纪,第七天休息,所以人们也就以七天为一个周期来安排自己的工作和生 活,而星期日是休息日。从实际的角度来讲,以七天为一个周期,长短也比较合适。所以尽管中国的传统工作周期是十天(比如王勃《滕王阁序》中说的“十旬休暇”,即是指官员的工作每十日为一个周期,第十日休假),但后来也采取了西方的星期制度。

在日常生活中,我们常常遇到要知道某一天是星期几的问题。有时候,我们还想知 道历史上某一天是星期几。通常,解决这个方法的有效办法是看日历,但是我们总不会随时随身带着日历,更不可能随时随身带着几千年的万年历。假如是想在计算机编程中计算某一天是星期几,预先把一本万年历存进去就更不现实了。这时候是不是有办法通过什么公式,从年月日推出这一天是星期几呢?

答案是肯定的。其实我们也常常在这样做。我们先举一个简单的例子。比如,知道 了2004年5月1日是星期六,那么2004年5月31日“世界无烟日”是星期几就不难推算出来。我们可以掰着指头从1日数到31日,同时数星期,最后可以数出5月31日是星期一。

其实运用数学计算,可以不用掰指头。我们知道星期是七天一轮回的,所以5月1日是星期六,七天之后的5月8日也是星期六。在日期上,8-1=7,正是7的倍数。同样,5月15日、5月22日和5月29日也是星期六,它们的日期和5月1日的差值分别是14、21和28,也都是7的倍数。那么5月31日呢?31-1=30,虽然不是7的倍数,但是31除以7,余数为2,这就是说,5月31日的星期,是在5月1日的星期之后两天。星期六之后两天正是星期一。这个简单的计算告诉我们计算星期的一个基本思路:首先,先要知道在想算的日子之前的一个确定的日子是星期几,拿这一天做为推算的标准,也就是相当于一个计算的“原点”。其次,知道想算的日子和这个确定的日子之间相差多少天,用7除这个日期的差值,余数就表示想算的日子的星期在确定的日子的星期之后多少天。如果余数是0,就表示这两天的星期相同。显然,如果把这个作为“原点”的日子选为星期日,那么余数正好就等于星期几,这样计算就更方便了。

但是直接计算两天之间的天数,还是不免繁琐。比如1982年7月29日和2004年5月1日之间相隔7947天,就不是一下子能算出来的。

它包括三段时间:

一,1982年7月29日以后这一年的剩余天数;

二,1983-2003这二十一个整年的全部天数;

三,从2004年元旦到5月1日经过的天数。

第二段比较好算,它等于21*365+5=7670天,之所以要加5,是因为这段时间内有5个闰年。第一段和第三段就比较麻烦了,比如第三段,需要把5月之前的四个月的天数累加起来,再加上日期值,即31+29+31+30+1=122天。同理,第一段需要把7月之后的五个月的天数累加起来,再加上7月剩下的天数,一共是155天。所以总共的相隔天数是122+7670+155=7947天。

仔细想想,如果把“原点”日子的日期选为12月31日,那么第一段时间也就是一个整年,这样一来,第一段时间和第二段时间就可以合并计算,整年的总数正好相当于两个日子的年份差值减一。如果进一步把“原点”日子选为公元前1年12月31日(或者天文学家所使用的公元0年12月31日),这个整年的总数就正好是想算的日子的年份减一。这样简化之后,就只须计算两段时间:一,这么多整年的总天数;二,想算的日子是这一年的第几天。巧的是,按照公历的年月设置,这样反推回去,公元前1年12月31日正好是星期日,也就是说,这样算出来的总天数除以7的余数正好是星期几。那么现在的问题就只有一个:这么多整年里面有多少闰年。这就需要了解公历的置闰规则了。

我们知道,公历的平年是365天,闰年是366天。置闰的方法是能被4整除的年份在2月加一天,但能被100整除的不闰,能被400整除的又闰。因此,像1600、2000、2400年都是闰年,而1700、1800、1900、2100年都是平年。公元前1年,按公历也是闰年。

因此,对于从公元前1年(或公元0年)12月31日到某一日子的年份Y之间的所有整年中的闰年数,就等于

[(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400],[...]表示只取整数部分。第一项表示需要加上被4整除的年份数,第二项表示需要去掉被100整除的年份数,第三项表示需要再加上被400整除的年份数。之所以Y要减一,这样,我们就得到了第一个计算某一天是星期几的公式:

W = (Y-1)*365 + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + D. (1)

其中D是这个日子在这一年中的累积天数。算出来的W就是公元前1年(或公元0年)12月31日到这一天之间的间隔日数。把W用7除,余数是几,这一天就是星期几。比如我们来算2004年5月1日:

W = (2004-1)*365 + [(2004-1)/4] - [(2004-1)/100] + [(2004-1)/400] +

(31+29+31+30+1)= 731702,

731702 / 7 = 104528……6,余数为六,说明这一天是星期六。这和事实是符合的。

上面的公式(1)虽然很准确,但是计算出来的数字太大了,使用起来很不方便。仔细想想,其实这个间隔天数W的用数仅仅是为了得到它除以7之后的余数。这启发我们是不是可以简化这个W值,只要找一个和它余数相同的较小的数来代替,用数论上的术语来说,就是找一个和它同余的较小的正整数,照样可以计算出准确的星期数。

显然,W这么大的原因是因为公式中的第一项(Y-1)*365太大了。其实,

(Y-1)*365 = (Y-1) * (364+1)

= (Y-1) * (7*52+1)

= 52 * (Y-1) * 7 + (Y-1),

这个结果的第一项是一个7的倍数,除以7余数为0,因此(Y-1)*365除以7的余数其实就

等于Y-1除以7的余数。这个关系可以表示为:

(Y-1)*365 ≡ Y-1 (mod 7).

其中,≡是数论中表示同余的符号,mod 7的意思是指在用7作模数(也就是除数)的情况下≡号两边的数是同余的。因此,完全可以用(Y-1)代替(Y-1)*365,这样我们就得到了那个著名的、也是最常见到的计算星期几的公式:

W = (Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + D. (2)

这个公式虽然好用多了,但还不是最好用的公式,因为累积天数D的计算也比较麻烦。是不是可以用月份数和日期直接计算呢?答案也是肯定的。我们不妨来观察一下各个月的日数,列表如下:

月 份:1月 2月 3月 4月 5月 6月 7月 8月 9月 10月 11月 12月

--------------------------------------------------------------------------

天 数: 31 28(29) 31 30 31 30 31 31 30 31 30 31

如果把这个天数都减去28(=4*7),不影响W除以7的余数值。这样我们就得到另一张表:

月 份:1月 2月 3月 4月 5月 6月 7月 8月 9月 10月 11月 12月

------------------------------------------------------------------------

剩余天数: 3 0(1) 3 2 3 2 3 3 2 3 2 3

平年累积: 3 3 6 8 11 13 16 19 21 24 26 29

闰年累积: 3 4 7 9 12 14 17 20 22 25 27 30

仔细观察的话,我们会发现除去1月和2月,3月到7月这五个月的剩余天数值是3,2,3,2,3;8月到12月这五个月的天数值也是3,2,3,2,3,正好是一个重复。相应的累积天数中,后一月的累积天数和前一月的累积天数之差减去28就是这个重复。正是因为这种规律的存在,平年和闰年的累积天数可以用数学公式很方便地表达:

╭ d; (当M=1)

D = { 31 + d; (当M=2) (3)

╰ [ 13 * (M+1) / 5 ] - 7 + (M-1) * 28 + d + i. (当M≥3)

其中[...]仍表示只取整数部分;M和d分别是想算的日子的月份和日数;平年i=0,闰年i=1。对于M≥3的表达式需要说明一下:[13*(M+1)/5]-7算出来的就是上面第二个表中的平年累积值,再加上(M-1)*28就是想算的日子的月份之前的所有月份的总天数。这是一个很巧妙的办法,利用取整运算来实现3,2,3,2,3的循环。比如,对2004年5月1日,有:

D = [ 13 * (5+1) / 5 ] - 7 + (5-1) * 28 + 1 + 1

= 122,

这正是5月1日在2004年的累积天数。假如,我们再变通一下,把1月和2月当成是上一年的“13月”和“14月”,不仅仍然符合这个公式,而且因为这样一来,闰日成了上一“年”(一共有14个月)的最后一天,成了d的一部分,于是平闰年的影响也去掉了,公式就简化成:

D = [ 13 * (M+1) / 5 ] - 7 + (M-1) * 28 + d. (3≤M≤14) (4)

上面计算星期几的公式,也就可以进一步简化成:

W = (Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + [ 13 * (M+1) / 5 ] - 7+ (M-1) * 28 + d.

因为其中的-7和(M-1)*28两项都可以被7整除,所以去掉这两项,W除以7的余数不变,公式变成:

W = (Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + [ 13 * (M+1) / 5 ] + d.(5)

当然,要注意1月和2月已经被当成了上一年的13月和14月,因此在计算1月和2月的日子的星期时,除了M要按13或14算,年份Y也要减一。比如,2004年1月1日是星期四,用这个公式来算,有:

W = (2003-1) + [(2003-1)/4] - [(2003-1)/100] + [(2003-1)/400] + [13*(13+1)/5]+ 1

= 2002 + 500 - 20 + 5 + 36 + 1

= 2524;

2524 / 7 = 360……4.这和实际是一致的。

公式(5)已经是从年、月、日来算星期几的公式了,但它还不是最简练的,对于年份的处理还有改进的方法。我们先来用这个公式算出每个世纪第一年3月1日的星期,列表如下:

年份: 1(401,801,…,2001) 101(501,901,…,2101)

--------------------------------------------------------------------

星期: 4 2

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

年份:201(601,1001,…,2201) 301(701,1101,…,2301)

--------------------------------------------------------------------

星期: 0 5

可以看出,每隔四个世纪,这个星期就重复一次。假如我们把301(701,1101,…,2301)年3月1日的星期数看成是-2(按数论中对余数的定义,-2和5除以7的余数相同,所以可以做这样的变换),那么这个重复序列正好就是一个4,2,0,-2的等差数列。据此,我们可以得到下面的计算每个世纪第一年3月1日的星期的公式:

W = (4 - C mod 4) * 2 - 4. (6)

式中,C是该世纪的世纪数减一,mod表示取模运算,即求余数。比如,对于2001年3月1日,C=20,则:

W = (4 - 20 mod 4) * 2 - 4

= 8 - 4

= 4.

把公式(6)代入公式(5),经过变换,可得:

(Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] ≡ (4 - C mod 4) * 2 - 1 (mod 7). (7)

因此,公式(5)中的(Y-1) + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400]这四项,在计算每个世纪第一年的日期的星期时,可以用(4 - C mod 4) * 2 - 1来代替。这个公式写出来就是:

W = (4 - C mod 4) * 2 - 1 + [13 * (M+1) / 5] + d. (8)

有了计算每个世纪第一年的日期星期的公式,计算这个世纪其他各年的日期星期的公式就很容易得到了。因为在一个世纪里,末尾为00的年份是最后一年,因此就用不着再考虑“一百年不闰,四百年又闰”的规则,只须考虑“四年一闰”的规则。仿照由公式(1)简化为公式(2)的方法,我们很容易就可以从式(8)得到一个比公式(5)更简单的计算任意一天是星期几的公式:

W = (4 - C mod 4) * 2 - 1 + (y-1) + [y/4] + [13 * (M+1) / 5] + d. (9)

式中,y是年份的后两位数字。

如果再考虑到取模运算不是四则运算,我们还可以把(4 - C mod 4) * 2进一步改写成只含四则运算的表达式。因为世纪数减一C除以4的商数q和余数r之间有如下关系:

4q + r = C,

其中r即是 C mod 4,因此,有:

r = C - 4q

= C - 4 * [C/4]. (10)

(4 - C mod 4) * 2 = (4 - C + 4 * [C/4]) * 2

= 8 - 2C + 8 * [C/4]≡ [C/4] - 2C + 1 (mod 7). (11)

把式(11)代入(9),得到:

W = [C/4] - 2C + y + [y/4] + [13 * (M+1) / 5] + d - 1. (12)

这个公式由世纪数减一、年份末两位、月份和日数即可算出W,再除以7,得到的余数是几就表示这一天是星期几,唯一需要变通的是要把1月和2月当成上一年的13月和14月,C和y都按上一年的年份取值。因此,人们普遍认为这是计算任意一天是星期几的最好的公式。这个公式最早是由德国数学家克里斯蒂安·蔡勒(Christian Zeller, 1822- 1899)在1886年推导出的,因此通称为蔡勒公式(Zeller’s Formula)。为方便口算,式中的[13 * (M+1) / 5]也往往写成[26 * (M+1) / 10]。

现在仍然让我们来算2004年5月1日的星期,显然C=20,y=4,M=5,d=1,代入蔡勒

公式,有:

W = [20/4] - 40 + 4 + 1 + [13 * (5+1) / 5] + 1 - 1

= -15.

注意负数不能按习惯的余数的概念求余数,只能按数论中的余数的定义求余。为了方便计算,我们可以给它加上一个7的整数倍,使它变为一个正数,比如加上70,得到55。再除以7,余6,说明这一天是星期六。这和实际是一致的,也和公式(2)计算所得的结果一致。

最后需要说明的是,上面的公式都是基于公历(格里高利历)的置闰规则来考虑的。对于儒略历,蔡勒也推出了相应的公式是:

W = 5 - C + y + [y/4] + [13 * (M+1) / 5] + d - 1. (13)

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

(2005-10-20 22:25:00) --------(4575252)

计算任何一天是星期几的几种算法近日在论坛上看到有人在问星期算法,特别整理了一下,这些算法都是从网上搜索而来,算法的实现是我在项目中写的。希望对大家有所帮助。

1:常用公式

W = [Y-1] + [(Y-1)/4] - [(Y-1)/100] + [(Y-1)/400] + D

Y是年份数,D是这一天在这一年中的累积天数,也就是这一天在这一年中是第几天。

2:蔡勒(Zeller)公式

w=y+[y/4]+[c/4]-2c+[26(m+1)/10]+d-1

公式中的符号含义如下,w:星期;c:世纪;y:年(两位数); m:月(m大于等于3,小于等于14,即在蔡勒公式中,某年的1、2月要看作上一年的13、14月来计算,比如2003年1月1日要看作2002年的13月1日来计算);d:日;[ ]代表取整,即只要整数部分。相比于通用通用计算公式而言,蔡勒(Zeller)公式大大降低了计算的复杂度。

3:对蔡勒(Zeller)公式的改进

相比于另外一个通用通用计算公式而言,蔡勒(Zeller)公式大大降低了计算的复杂度。不过,笔者给出的通用计算公式似乎更加简洁(包括运算过程)。现将公式列于其下:

W=[y/4]+r (y/7)-2r(c/4)+m’+d

公式中的符号含义如下,r ( )代表取余,即只要余数部分;m’是m的修正数,现给出1至12月的修正数1’至12’如下:(1’,10’)=6;(2’,3’,11’)=2;(4’,7’)=5;5’=0;6’=3;8’=1;(9’,12’)=4(注意:在笔者给出的公式中,y为润年时1’=5;2’=1)。其他符号与蔡勒(Zeller)公式中的含义相同。

4:基姆拉尔森计算公式

W= (d+2*m+3*(m+1)/5+y+y/4-y/100+y/400) mod 7

在公式中d表示日期中的日数,m表示月份数,y表示年数。

注意:在公式中有个与其他公式不同的地方:

把一月和二月看成是上一年的十三月和十四月,例:如果是2004-1-10则换算成:2003-13-10来代入公式计算。

8.测试结果

通过纯DOS界面显示调试结果,可以显示公农历日期、星期、节气,天干地支。

测试结果图:

9.部分源代码

#include<iostream>

#include<iomanip>

#include<fstream>

#include<string>

using namespace std;

#define FIRSTYEAR 1936 /* 计算起点年*/

#define LASTYEAR 2031 /* 计算末点年*/

SolarShow(int ,int ,string);

week(int,int);

static int AccDays[2][15] ={{ 0,0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334,365,396},

{0, 0,31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366, 397}};

void main()

{

int Year; //公历年

int Month;//公历月

int cCmd,c;

char buf[80];

string sFile="D:\\文件\\rili.txt";

begin:

cout<<"\n\n\t 简易万年历\n\t=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=";

cout <<"\n\t (1) 日 历 查 询 (2) 退 出 系 统";

cout<<"\n\t=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=";

cout<<"\n\t请 输 入 您 的 选 择:";

begin1:

cin>>cCmd;

c=cin.rdstate();

while(c)

{cin.clear();cin.getline(buf,80);

cout<<"\n\t非法输入 !重新输入:";

cin>>cCmd;

c=cin.rdstate();}

if(cCmd!=1&&cCmd!=2)

{cout<<"\n\t错误选择,重新选择:";goto begin1;}

if(cCmd==2)

{

cout<<"\n\t\t 操作完毕,数据已保存到"<<sFile;

cout<<"\n\t=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=";

cout<<"\n\t 请按任意键退出系统!";

cout<<"\n\t=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*=*="<<endl;

return;

}

if(cCmd==1)

{

char cchoice;

do

{

cout<<"\n\t默认路径是:D:\\文件\\rili.txt\n\t请输入存盘路径:";

cin>>sFile; cin.getline(buf,100);

cout<<"\t确认无误是"<<sFile<<"吗?是(Y)否(N):";

cin>>cchoice;

}while(cchoice!='y'&&cchoice!='Y');

cout<< "\t-----------------\n\t请输入公历年 ,月份(用空格格开):";

cin >>Year;cin>>Month;

SolarShow(Year,Month,sFile);

strat:

cout<<"\n\t===*=======*=======*=======*=======*=======*=======*===="; cout<<"\n\t (1)上一月 (3)上一年 (5)重新输入年月\n";

cout<<"\n\t (2)下一月 (4)下一年 (6)返回上级菜单";

cout<<"\n\t===*=======*=======*=======*=======*=======*=======*===="; cout<<"\n\t请 输 入 您 的 选 择:";

cin>>cCmd;

c=cin.rdstate();

while(c)

{cin.clear();cin.getline(buf,80);

cout<<"\n\t非法输入 !重新输入:";

cin>>cCmd;

c=cin.rdstate();}

switch(cCmd)

{

case 3: { Year=Year-1;SolarShow(Year,Month,sFile);goto strat;}

case 1: { Month=Month-1;SolarShow(Year,Month,sFile);goto strat;}

case 4: { Year=Year+1;SolarShow(Year,Month,sFile);goto strat;}

case 2: { Month=Month+1;SolarShow(Year,Month,sFile);goto strat;}

case 5: {cout << "\n\t请输入公历年 ,月份(用空格格开):";

cin >>Year;cin>>Month;SolarShow(Year,Month,sFile);goto strat;}

case 6: {cout<<"\n\t 返 回 上 级 菜 单!";goto begin;}

default: {cout<<"\n\t错误选择!重新选择!\n";goto strat;}

}

}

}

1.闰年判断函数

int GetLeap( int year )

{ int leap;

if ((year%400==0)||(year%4==0 && year%100!=0)) return leap=1;

else return leap=0;

}*/

2.星期几计算

week(int year,int month)

{ int differ=year+(year-1)/4-(year-1)/100+(year-1)/400;

int leap;

if ((year%400==0)||(year%4==0 && year%100!=0)) leap=1;

else leap=0;

int acc=AccDays[leap][month];

differ=differ+acc;

int date=differ%7;

return date;}

3.日历显示程序

SolarShow(int SolarYear ,int SolarMonth,string sFile)

{

start:

int c=cin.rdstate();char buf[100];

while(c)

{

cin.clear();cin.getline(buf,80);

cout<<"\n\t非法输入 !重新输入:";

cin>>SolarYear;cin>>SolarMonth;

c=cin.rdstate();

}

//判断年月范围

while((SolarYear<FIRSTYEAR||SolarYear>LASTYEAR)&&(SolarMonth<1||SolarMonth>12))

//{do

{cout<<"\n\t出错!年份范围(1936—2031)";

cout<<"\n\t出错!月份范围(1—12)";

cout<<"\n\t重输年月份(用空格格开):";

cin>>SolarYear;cin>>SolarMonth;goto start;

//}while((SolarYear<FIRSTYEAR||SolarYear>LASTYEAR)&&(SolarMonth<1||SolarMonth>12));

}

while ((SolarYear<FIRSTYEAR)||(SolarYear>LASTYEAR))

//{do

{cout<<"\n\t出错!年份范围(1936—2031)";

cout<<"\n\t重输年月份(用空格格开):";

cin>>SolarYear;cin>>SolarMonth;goto start;

//}while((SolarYear<FIRSTYEAR)||(SolarYear>LASTYEAR));

}

while ((SolarMonth<1)||(SolarMonth>12))

//{do

{cout<<"\n\t出错!月份范围(1—12)";

cout<<"\n\t重输年月份(用空格格开):";

cin>>SolarYear;cin>>SolarMonth;goto start;

//}while((SolarMonth<1)||(SolarMonth>12));

}

static char *cLunarMonth[14]={"0","正月","二月","三月","四月","五月","六月","七月","八月","九月","十月","冬月","腊月","正月"};

/*农历日*/

static char *cSolarMonth[13]={"0","(1~19)山羊 (20~31)水瓶","(1~18)水瓶 (19~28)双鱼",

"(1~20)双鱼 (21~31)白羊","(1~20)白羊 (21~30)金牛",

"(1~20)金牛 (21~31)双子","(1~20)双子 (21~30)巨蟹",

"(1~20)巨蟹 (21~31)狮子","(1~21)狮子 (22~31)处女",

"(1~22)处女 (23~30)天秤","(1~22)天秤 (23~31)天蝎",

"(1~22)天蝎 (23~30)人马","(1~20)人马 (21~31)山羊"};

static char *cLunarDate[31]={"0","初一","初二","初三","初四","初五","初六","初七","初八","初九","初十","十一","十二","十三","十四","十五","十六","十七","十八","十九","二十","廿一","廿二","廿三","廿四","廿五","廿六","廿七","廿八","廿九","三十"};

static char *cSolarDate[32]={"0"," 1"," 2"," 3"," 4"," 5"," 6"," 7"," 8"," 9","10","11","12","13","14","15","16","17","18","19","20","21","22","23","24","25","26","27","28","29","30","31"};

static char *ganzhi[61]={"0","甲子","乙丑","丙寅","丁卯","戊辰","己巳","庚午","辛未","壬申","癸酉","甲戌","乙亥",

"丙子","丁丑","戊寅","己卯","庚辰","辛巳","壬午","癸未","甲申","乙酉","丙戌","丁亥",

"戊子","己丑","庚寅","辛卯","壬辰","癸巳","甲午","乙未","丙申","丁酉","戊戌","己亥",

"庚子","辛丑","壬寅","癸卯","甲辰","乙巳","丙午","丁未","戊申","己酉","庚戌","辛亥",

"壬子","癸丑","甲寅","乙卯","丙辰","丁巳","戊午","己未","庚申","辛酉","壬戌","癸亥"};

static char *Weekday[8]={"0","日","一","二","三","四","五","六"};

static int SolarPermonth[2][13] ={{0,31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },

{0,31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }};

//每月1日距离1月1日的天数

struct tagLunarCal

{

int BaseDays; /* 到西历 1 月 1 日到农历正月初一的累积日数 */

int Intercalation; /* 闰月月份. 0==此年没有闰月 */

int BaseGZ; /* 此年西历 1 月 1 日之干支序号减 1 */

int MonthDays[13]; /* 此农历年每月之大小, 0==小月(29日), 1==大月(30日) */

};

10.结束语

经历了这次课程设计的设计和制作的整个过程,我才发现我知识的贫乏和知识面的狭窄,原本以为不就是一个小小的课程设计,小case,很容易就可以搞定,可是到了后来实际的操作的时候可是花费了我九牛二虎之力,不说其他的,只说时间上面,我就花了一两个星期,更不用说做其他了。

这次作课程设计我也是收获很大,首先,我对C++语言有了更深的了解,使用起来也更加的熟练,原来有一些不知道和不熟悉的函数,我学会用了。其次,我这一次找了不少的资料,找专业的书籍、上网找各方面的资料,也使我学到了一些书本中学不到的知识,让我长了见识。同时我们同学之间也互相的探讨,也增进了我们同学之间的感情。

这次作的这一个设计,虽然我自己觉的很有的成就感,但是还有很多不足的地方,在以后的学习工作了还有待改进,相信以后我一定会做得更好的。

11.参考文献

1、丁元杰 《单片微机原理及应用》 北京机械工业出版社 1999

2、MICROSOFT 《Visual C++6.0 组件工具指南》 北京希望电子出版社 1999

3、张海藩 《软件工程导论》 清华大学出版社 2003

4、《信息控制 CAD 课程设计实验指导书》西南科技大学 2005

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多