摘要:使用C语言打印年历
最初看到这个是觉得比较有意思,记得读书时候只写过月历,年历要复杂一些,现在水平提高了,来重新写一下吧。
看了网上的一些代码,大都比较繁琐,本文采用全新的思路完成年历打印。
整体思路:
来看一下效果吧:
看了一下网上的代码,思路大都是一行一行打印,这样涉及到月份,星期以及月首的空格打印都非常复杂,导致代码量庞大,并且逻辑非常复杂,本文旨在使用全新的思路,使用较好理解的代码,以及较少的代码量完成年历打印。
先讲思路,我们知道,每月最多31天,打印出来5~6行(每行最多7个),传统方法复杂的原因在于只能逐行打印。本文使用一个21 X 24的二维数组(每个月占用7 X 6个空间),用来保存当月的日期,接下来上代码。
代码解释:
cal_first函数用来计算月首第一天是星期几,方便确定正确的打印位置。
int cal_first(int y, int m) { int d = 1; if(m<=2) { m += 12; y -= 1; } return (d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7 + 1; }
接下来是show_year函数,略微复杂。
void show_year() { int i,j,k,t,n; //用来辅助计数 int table[24][21] = {0}; int month_day[12] = {31,28,31,30,31,30,31,31,30,31,30,31}; if ((0 == year%4 || 0 == year%400) && 0 != year%100)//闰年检查 { month_day[2] = 29; } printf (" ******************\n"); printf (" %d \n", year); printf (" ******************\n"); for (i=0; i<12; i++) { month = i+1; n = cal_first(); for (j=(i/3)*6; j<(i/3)*6+6; j++) { for (k=(i%3)*7; k<(i%3)*7+7; k++) { t = (j%6)*7 + k%7 - n + 2; if (0 >= t) { continue; } if (t <= month_day[i]) { table[j][k] = t; } else { continue; } } } } for (i=0; i<24; i++) { if (0 == i) { printf (" Jan Feb Mar\n"); printf (" Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun\n"); } if (6 == i) { printf (" Apr May Jun\n"); printf (" Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun\n"); } if (12 == i) { printf (" Jul Agu Sep\n"); printf (" Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun\n"); } if (18 == i) { printf (" Oct Nov Dec\n"); printf (" Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun Mon Tue Wen Thu Fir Sat Sun\n"); } for (j=0; j<21; j++) { if (7 == j || 14 == j) { printf (" "); } if (0 == table[i][j]) { printf (" "); continue; } printf ("%4d", table[i][j]); } printf ("\n"); } }