ACM集训的1B。。。。黑色星期五。。。。2333333

题目:

印象中有好多个13号是星期五,13号在星期五比在其他日子少吗?为了回答这个问题,写一个程序,要求计算每个月的十三号落在周一到周日的次数。给出N年的一个周期,要求计算1900年1月1日至1900+N-1年12月31日中十三号落在周一到周日的次数,用现成的函数
请不要预先算好数据(就是叫不准打表)!

输入格式:
一个正整数n

输出格式:
七个在一行且相分开的整数,它们代表13日是星期六,星期日,星期一...星期五的次数..

样例输入
20
样例输出
36 33 34 33 35 35 34

 

 

 

思路:

这个是根据公式计算啦。比如:

日期格式 y-m-d,y为年份数(y>1582),m为月份数(0<m<13),d为日数(0<d<28、29、30、31)。y、m、d为整数。例如2008-8-1,各变量分别是y=2008,m=8,d=1。

1、常用公式

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

式中变量说明:W为星期数,y为年份数,D为该日期在该年中的排序数;[X]为对X取整,下同。

2、蔡勒(Zeller)公式

W=Y+[Y/4]+[C/4]-2C+[26(M+1)/10]+d-1

公式中的符号含义如下:

W为星期数;C为世纪;Y为年(两位数); M为月数(M=m(当m>2);M=m+12(m<3));d为日。

相比于通用通用计算公式而言,蔡勒(Zeller)公式大大降低了计算的复杂度。

3、基姆拉尔森计算公式

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来代入公式计算。

4、蔡勒公式一种改进

相比于另外一个通用通用计算公式而言,蔡勒(Zeller)公式大大降低了计算的复杂度。不过,另外有一个似乎更加简洁更简单的改进公式,提出这个公式的人是冯思琮[1] [2] :

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

下,其他符号与蔡勒公式中含义相同。r(X)为对表达式X取余;
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。
-----------------------------
[注] 以上的公式都只适合于1582年10月15日之后的情形,即我国明朝万历十年间。罗马教皇格里高利十三世在1582年组织了一批天文学家,根据哥白尼日心说计算出来的数据,对儒略历作了修改。将1582年10月5日到14日之间的10天宣布撤销,继10月4日之后为10月15日。
后来人们将这一新的历法称为“格里高利历”,也就是今天世界上所通用的历法,简称格里历或公历。
------------------------------------------------------------------------
以上公式本人没有验证过,不保证编辑过程中出错的可能性。=.=

 

 

代码:

 1 #include<iostream>
 2 using namespace std;
 3 int mon[12]={31,28,31,30,31,30,31,31,30,31,30,31};
 4 int day[7]={0,0,0,0,0,0,0};
 5 int main(){
 6        int k=6,n;                 //因为1900.1.1是星期六啦~\(≧▽≦)/~啦啦啦
 7     scanf("%d",&n);
 8      n+=1900;
 9      for(int i=1900;i<n;i++){
10       if((i%100!=0&&i%4==0)||i%400==0)mon[1]=29;  //闰年哟
11       for(int j=0;j<12;j++){
12            day[k]++;                  //这块是记录星期几分别有几天的数组
13            k+=mon[j];                //每次相当于把k变成下个月的13号
14            k%=7;                       //取余数方便算星期几
15       }
16       mon[1]=28;                        //平年的情况~
17      }
18      printf("%d",day[6]);
19      for(int i=0;i<6;i++)printf(" %d",day[i]);
20 }

 

posted @ 2016-07-01 13:51  meetviolet  Views(546)  Comments(0Edit  收藏  举报