C++数组在年历打印中的运用
我家小朋友正在学C++数组,所以呢自己就边学边写点体会给他看,就以一个打印年历的实例帮他更好地理解和掌握数组。刚好在2021年元旦写完,现发出来与C初学者分享一下,有误之处敬请谅解。
一维数组的定义:
声明一个数组的格式: type array[size];
#include <iomanip.h> int main(void) { int a[5]={1,2,3,4,5}; char b[]="hello"; cout<<sizeof(int)<<endl; //变量int占用的比特数 cout<<sizeof(a)<<endl; //数组a[]占用的比特数 cout<<"数组a的元素个数:"<<sizeof(a)/sizeof(int)<<endl; cout<<sizeof(char)<<endl; //变量int占用的比特数 cout<<sizeof(b)<<endl; //数组b[]占用的比特数 cout<<"数组b的元素个数:"<<sizeof(b)/sizeof(char)<<endl; for (i=0;i<6;i++) cout<<b[i]<<"|"; cout<<endl; //打印数组b[],最后一个字符是'\0',字符串结束符。 return 0; }
运行结果:
注1:结束符在定义中不写出来的,但也可以写成:
char b[]={"hello"}; //不省掉{} char b[]={‘h’,’e’,’l’,’l’,’o’,’\0’};
看上图运行结果,’\0’打印出来是空的,但它不是空格‘ ’。
注2:数组的下标从0开始计!
int a[5]={1,2,3,4,5};
即:
int a[5]; a[0]=1; a[1]=2; a[2]=3; a[3]=4; a[4]=5; //最后一个是a[4] int b[6]= "hello";
即:
int b[6]; a[0]=’h’; a[1]=’e’; b[2]=’l’; …; b[5]=’\0’; //最后一个是b[5]
也就是array[size]的最后一个元素是array[size-1]。
数组可以在声明时直接赋值,也可以一个一个地赋值。
如果数组的值有某种规律就可以用循环来赋值:
int a[30]; for (i=0;i<30;i++) a[i]=i+1; //把1~30赋值给a[0]~a[29]
二维数组的定义:
如下,定义一个3行5列的整型二维数组int a[3][5];
它对应的元素为:
a[0][0] a[0][1] a[0][2] a[0][3] a[0][4]
a[1][0] a[1][1] a[1][2] a[1][3] a[1][4]
a[2][0] a[2][1] a[2][2] a[2][3] a[2][4]
定义完数组后,一般就用二重循环来给它赋值。
#include <iomanip.h> int main(void) { int a[3][5]; int i,j; //赋值 for (i=0;i<3;i++) for (j=0;j<5;j++){ a[i][j]=j+1+i*5; } //列表 for (i=0;i<3;i++) for (j=0;j<5;j++){ cout<<setw(5)<<a[i][j]; if (j==4) cout<<endl; } //setw(x)函数设置输出的宽度,并且右对齐。 }
运行结果如下:
下来我们定义一个6行7列的二维数组存放每个月的日期,为什么是6行,看下面系统时间的截图就知道了。
二维数组的赋值与输出:
#include <iomanip.h> int main(void) { int month[6][7]; int i, j; //赋值 for (i=0;i<6;i++) for (j=0;j<7;j++) month[i][j]=j+1+i*7; //列表 for (i=0;i<6;i++) for (j=0;j<7;j++){ cout<<setw(3)<<month[i][j]; if (j==6) cout<<endl; } return 0; }
运行结果如下:
每月的一号不可能都是星期一,所以我们用基姆拉尔森公式来定义一个星期函数WeekDay(),用它来计算每月第一天的星期数。
int WeekDay(int y,int m,int d) { if(m<3)--y,m+=12; //一、二月看作上年的13、14月 return(d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7+1; }
函数返回整数1~7,对应星期一到星期日。
#include <iomanip.h> int WeekDay(int y,int m,int d) { if(m<3)--y,m+=12; return(d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7+1; } int main(void) { int month[6][7]; int i, j; int w; w = WeekDay(2020,1,1); //计算2020年元旦的星期数。 //赋值 for (i=0;i<6;i++) for (j=0;j<7;j++){ month[i][j]=j+1+i*7; month[i][j]-=w-1; //赋值后减去w再加1。 // 上两行即month[i][j]=j+1+i*7-w+1; } //列表 for (i=0;i<6;i++) for (j=0;j<7;j++){ cout<<setw(3)<<month[i][j]; if (j==6) cout<<endl; } return 0; }
运行结果如下:
正式使用时,把小于1和大于月底日期的数字不显示即可。所以,又要增加一个数组存放一年12个月的天数。
int days[12]={31,28,31,30,31,30,31,31,30,31,30,31};
另还要判断是否闰年,如闰年则置二月天数days[1]为29。
• int y = 2000; • if (y%4==0&&y%100!=0||y%400==0) days[1]=29;
输出指定年份年历表的代码如下:
#include <iomanip.h> #include <stdlib.h> //调用DOS清屏命令cls int WeekDay(int y,int m,int d) { if(m<3)--y,m+=12; return(d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7+1; } int main(void ) { char *s[]={"一","二","三","四","五","六","日","\n"}; //表头 int days[12]={31,28,31,30,31,30,31,31,30,31,30,31}; int i,j,k,y,w; int month[6][7]; for(;;){ cout<<"输入年份(0退出):"; cin>>y; system("CLS"); cout<<"你输入的年份是:"<<y<<endl; if (y==0) return 0; if (y%4==0&&y%100!=0||y%400==0) days[1]=29; for (k=0;k<12;k++){ for (i=0;i<6;i++) for (j=0;j<7;j++){ w = WeekDay(y,k+1,1); //注意月份是k+1 month[i][j]=j+1+i*7; month[i][j]-=w-1; //赋值后减去w再加1。 } cout<<k+1<<"月:"<<endl; //注意月份是k+1 for(i=0;i<8;i++) cout<<setw(3)<<s[i]; //打印星期表头 for (i=0;i<6;i++) for (j=0;j<7;j++){ if (month[i][j]>days[k]) break; if (month[i][j]>0) //大于0输出,反之输出3个空格 cout<<setw(3)<<month[i][j]; else cout<<" "; if (j==6&&month[i][j]!=days[k]) cout<<endl; //最后一列输出回车,但月底一天刚好周日则不用回车 } cout<<endl; }//next k days[1]=28; //因多次用到days[],置平月天数千万不能省略 cout<<endl; } }//end main();
运行结果如下:
体会一下源代码中何时添加回车,何时输出空格,都用if来判断输出。
接下来就做双排的年历,这时数组需要6行14列,可以看作并排了两个7列的表格,用2个循环分别来赋值。
#include <iomanip.h> int main(void) { int i,j; int month[6][14]; for (i=0;i<6;i++) for (j=0;j<7;j++) month[i][j]=j+1+i*7; for (i=0;i<6;i++) for (j=7;j<14;j++) month[i][j]=j-6+i*7; for (i=0;i<6;i++) for (j=0;j<14;j++){ cout<<setw(3)<<month[i][j]; if (j==6) cout<<" | "; //两个表格用|分隔 if (j==13) cout<<endl; } return 0; } //end main()
把上面的置数部分的用了2个循环,可以从函数main()中拿出来另外写成一个函数initMon()用来初始化数组。此时二维数组month[][]是全局变量要放在所有函数的前面。源代码如下:
#include <iomanip.h> int month[6][14]; void initMon(void) { int i,j; for (i=0;i<6;i++) for (j=0;j<7;j++) month[i][j]=j+1+i*7; for (i=0;i<6;i++) for (j=7;j<14;j++) month[i][j]=j-6+i*7; } //end initMon() int main(void) { int i,j; initMon(); for (i=0;i<6;i++) for (j=0;j<14;j++){ cout<<setw(3)<<month[i][j]; if (j==6) cout<<" | "; if (j==13) cout<<endl; } return 0; } //end main()
执行结果如下:
上图中并排两个月份的首日需要调用WeekDay()让它们分别移位,所以初始化函数initMonth()要引进2个月份参数;另外新加了一个三目运算符(true or false)?x:y,条件为真取值x反之取值y。
void initMonth(int m1, int m2) { int i,j,w1,w2; w1=WeekDay(year,m1,1); w2=WeekDay(year,m2,1); days[1] = (year%4==0&&year%100!=0||year%400==0) ? 29:28; for (i=0;i<6;i++) for (j=0;j<7;j++){ month[i][j]=j+1+i*7; month[i][j]-=w1-1; if (month[i][j]>days[m1-1]) month[i][j]=0; } for (i=0;i<6;i++) for (j=7;j<14;j++){ month[i][j]=j-6+i*7; month[i][j]-=w2-1; if (month[i][j]>days[m2-1]) month[i][j]=0; } }
初始化月份数组后,用多重循环控制行列来输出各个月份,输出时把小于1的数字都用空格替换。月份有两种排列,横排或竖排。特别要注意的是把握好循环变量与各行各列的月份数、日期数之间的函数关系。完成源代码如下:
#include <iostream> #include <iomanip> #include <process.h> //调用system()函数 using namespace std; int year; int month[6][14]; int days[12]={31,28,31,30,31,30,31,31,30,31,30,31}; int WeekDay(int y, int m, int d) { if(m<3)--y,m+=12; //1、2月看作上年的13、14月 return(d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7+1; //基姆拉尔森公式,返回1~7对应星期一~星期日 } void initMonth(int m1, int m2) { int i,j,w1,w2; w1=WeekDay(year,m1,1); w2=WeekDay(year,m2,1); days[1]=(year%4==0&&year%100!=0||year%400==0)?29:28; for (i=0;i<6;i++) for (j=0;j<7;j++){ month[i][j]=j+1+i*7; month[i][j]-=w1-1; if (month[i][j]>days[m1-1]) month[i][j]=0; } for (i=0;i<6;i++) for (j=7;j<14;j++){ month[i][j]=j-6+i*7; month[i][j]-=w2-1; if (month[i][j]>days[m2-1]) month[i][j]=0; } //大于月底日期的数字都置为0; //每月第一天之前的都设成小于或等于0 } void Row0Col(void) { system("cls"); cout<<"你输入的年份是:"<<year<<endl<<endl; cout<<"你要竖向排列(1)还是横向排列(2)?(1 or 2)"; } int main(void) { int i,j,k,r=0; const char *s[]={"一","二","三","四","五","六","日","\n"}; for(;;){ cout<<"输入年份(0退出):"; do { cin>>year; if (year<1900){ system("cls"); cout<<"重新输入年份(年份数>=1900,0则退出):"; } if (year==0) return 0; } while(year<1900); Row0Col(); do{ cin>>r; if(r!=1||r!=2) Row0Col(); if(r==1||r==2) cout<<r; }while(!(r==1||r==2)); if (r==1){ //竖向排列 cout<<"\n\n竖向排列:"<<endl; for (k=0;k<6;k++){ cout<<k+1<<"月:"; for (i=0;i<6;i++) cout<<" "; cout<<k+7<<"月:"<<endl; for(i=0;i<7;i++) cout<<" "<<s[i]; cout<<" "; for(i=0;i<8;i++) cout<<" "<<s[i]; initMonth(k+1,k+7); for (i=0;i<6;i++){ for (j=0;j<14;j++){ //大于0的输出日期,否则输出空格 if (month[i][j]>0) cout<<setw(3)<<month[i][j]; else cout<<" "; if (j==6) cout<<" | "; if (j==13) cout<<endl; }//next j }//next i }//next k } //else if(r==2) else{ //横向排列 cout<<"\n\n横向向排列:"<<endl; for (k=0;k<6;k++){ cout<<k*2+1<<"月:"; for (i=0;i<6;i++) cout<<" "; cout<<k*2+2<<"月:"<<endl; for(i=0;i<7;i++) cout<<" "<<s[i]; cout<<" "; for(i=0;i<8;i++) cout<<" "<<s[i]; initMonth(k*2+1,k*2+2); for (i=0;i<6;i++){ for (j=0;j<14;j++){ if (i==5&&month[5][0]==0&&month[5][7]==0) break; //屏蔽空行 if (month[i][j]>0) cout<<setw(3)<<month[i][j]; else cout<<" "; //大于0的输出日期,否则输出空格 if (j==6) cout<<" | "; if (j==13) cout<<endl; }//next j }//next i }//next k }//end if-else cout<<endl; }//end for(;;) }//end main();
运行结果如下:
代码稍作修改,即可得到每一行排列3个月的输出。月份数组变更为6行21列,initMonth()同时初始化三个月份,同时调整一下循环变量与月份数组下标的关系。完整代码如下:
#include <iostream> #include <iomanip> #include <process.h> //调用system()函数 using namespace std; int year; int month[6][21]; int days[12]={31,28,31,30,31,30,31,31,30,31,30,31}; int WeekDay(int y, int m, int d) { if(m<3)--y,m+=12; //1、2月看作上年的13、14月 return(d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7+1; //基姆拉尔森公式,返回1~7对应星期一~星期日 } void initMonth(int m1, int m2, int m3) { int i,j,w1,w2,w3; w1=WeekDay(year,m1,1); w2=WeekDay(year,m2,1); w3=WeekDay(year,m3,1); days[1]=(year%4==0&&year%100!=0||year%400==0)?29:28; for (i=0;i<6;i++) for (j=0;j<7;j++){ month[i][j]=j+1+i*7; month[i][j]-=w1-1; if (month[i][j]>days[m1-1]) month[i][j]=0; } for (i=0;i<6;i++) for (j=7;j<14;j++){ month[i][j]=j-6+i*7; month[i][j]-=w2-1; if (month[i][j]>days[m2-1]) month[i][j]=0; } for (i=0;i<6;i++) for (j=14;j<21;j++){ month[i][j]=j-13+i*7; month[i][j]-=w3-1; if (month[i][j]>days[m3-1]) month[i][j]=0; } } void Row0Col(void) { system("cls"); cout<<"你输入的年份是:"<<year<<endl<<endl; cout<<"你要竖向排列(1)还是横向排列(2)?(1 or 2)"; } int main(void) { int i,j,k,r=0; const char *s[]={"一","二","三","四","五","六","日","\n"}; for(;;){ cout<<"输入年份(0退出):"; do { cin>>year; if (year<1900){ system("cls"); cout<<"重新输入年份(年份数>=1900,0则退出):"; } if (year==0) return 0; } while(year<1900); Row0Col(); do{ cin>>r; if(r!=1||r!=2) Row0Col(); if(r==1||r==2) cout<<r; }while(!(r==1||r==2)); if (r==1){ //竖向排列 cout<<"\n\n竖向排列:"<<endl; for (k=0;k<4;k++){ for(j=0;j<2;j++){ cout<<k+j*4+1<<"月:"; for(i=0;i<19;i++) cout<<" "; } cout<<k+9<<"月:"<<endl; for(j=0;j<2;j++){ for(i=0;i<7;i++) cout<<" "<<s[i]; cout<<" "; } for(i=0;i<8;i++) cout<<" "<<s[i]; initMonth(k+1,k+5,k+9); for (i=0;i<6;i++){ for (j=0;j<21;j++){ if (i==5 && month[5][0]==0 && month[5][7]==0 && month[5][14]==0) break; //屏蔽空行 if (month[i][j]>0) cout<<setw(3)<<month[i][j]; else cout<<" "; //大于0的输出日期,否则输出空格 if (j==6||j==13) cout<<" | "; if (j==20) cout<<endl; }//next j }//next i }//next k } //else if(r==2) else{ //横向排列 cout<<"\n\n横向排列:"<<endl; for (k=0;k<4;k++){ for(j=0;j<2;j++){ cout<<k*3+j+1<<"月:"; for(i=0;i<19;i++) cout<<" "; } cout<<k*3+3<<"月:"<<endl; for(j=0;j<2;j++){ for(i=0;i<7;i++) cout<<" "<<s[i]; cout<<" "; } for(i=0;i<8;i++) cout<<" "<<s[i]; initMonth(k*3+1,k*3+2,k*3+3); for (i=0;i<6;i++){ for (j=0;j<21;j++){ if (i==5 && month[5][0]==0 && month[5][7]==0 && month[5][14]==0) break; //屏蔽空行 if (month[i][j]>0) cout<<setw(3)<<month[i][j]; else cout<<" "; //大于0的输出日期,否则输出空格 if (j==6||j==13) cout<<" | "; if (j==20) cout<<endl; }//next j }//next i }//next k }//end if-else cout<<endl; }//end for(;;) }//end main();
运行结果如下:
三维数组的定义:
int month[6][7][12];
可以把它看作12个二维数组month[6][7]联立在一起,想像成一个12页的PPT文档,每页上都有一个6行7列的表格。
最后,我们来用三维数组来输出年历,先用一个三重循环来初始化全年的日期,然后按照排版的需求来输出这个三维数组。
先看一下未屏蔽掉数组中非正整数的代码:
#include <iostream> #include <iomanip> using namespace std; int month[6][7][12]; int days[12]={31,28,31,30,31,30,31,31,30,31,30,31}; int WeekDay(int y, int m, int d) { if(m<3)--y,m+=12; //1、2月看作上年的13、14月 return(d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7+1; //基姆拉尔森公式,返回1~7对应星期一~星期日 } void initMonth(int year) { int i,j,m,w; const char *s[]={"一","二","三","四","五","六","日","\n"}; days[1]=(year%4==0&&year%100!=0||year%400==0)?29:28; for (m=0;m<12;m++){ w=WeekDay(year,m+1,1); for (i=0;i<6;i++){ for (j=0;j<7;j++){ month[i][j][m]=j+1+i*7; month[i][j][m]-=w-1; if (month[i][j][m]>days[m]) month[i][j][m]=0; } } } } int main(void) { int i,j,m; initMonth(2021); cout<<2021<<"年:"<<endl; for (m=0;m<12;m++){ cout<<m+1<<"月:"<<endl; for(i=0;i<8;i++) cout<<" "<<s[i]; for (i=0;i<6;i++){ for (j=0;j<7;j++){ cout<<setw(3)<<month[i][j][m]; if (j==6) cout<<endl; } } } }//end main();
运行结果如下:
用了三维数组后一年12个月的日期初始化一次完成,再用再几个循环控制输出即可。3列横排的代码如下:
#include <iostream> #include <iomanip> using namespace std; #include <process.h> //调用system()函数 int month[6][7][12]; int days[12]={31,28,31,30,31,30,31,31,30,31,30,31}; int WeekDay(int y, int m, int d) { if(m<3)--y,m+=12; //1、2月看作上年的13、14月 return(d+2*m+3*(m+1)/5+y+y/4-y/100+y/400)%7+1; //基姆拉尔森公式,返回1~7对应星期一~星期日 } void initMonth(int year) { int i,j,m,w; days[1]=(year%4==0&&year%100!=0||year%400==0)?29:28; for (m=0;m<12;m++){ w=WeekDay(year,m+1,1); for (i=0;i<6;i++){ for (j=0;j<7;j++){ month[i][j][m]=j+1+i*7; month[i][j][m]-=w-1; if (month[i][j][m]>days[m]) month[i][j][m]=0; } } } } int main(void) { int i,j,k,w,m,year; const char *s[]={"一","二","三","四","五","六","日"}; for(;;){ cout<<"输入年份(0退出):"; do { cin>>year; if (year<1900){ system("cls"); cout<<"重新输入年份(年份数>=1900,0则退出):"; } if (year==0) return 0; } while(year<1900); system("cls"); cout<<"你输入的年份是:"<<year<<endl<<endl; initMonth(year); cout<<year<<"年:"<<endl; for (m=0;m<4;m++){ for (j=0;j<3;j++){ if (j==0) cout<<endl; cout<<m*3+j+1<<"月:"; for (i=0;i<19;i++) cout<<" "; if (m*3+j+1>9) cout<<"\b"; //修正2位数月份的宽度,"\b"为退格键BackSpace } cout<<endl; for (j=0;j<3;j++) for (i=0;i<7;i++){ cout<<" "<<s[i]; if (i==6&&j!=2) cout<<" | "; } cout<<endl; for (i=0;i<6;i++){ for (k=0;k<3;k++){ if (i==5 && month[5][0][m*3]==0 && month[5][0][m*3+1]==0 && month[5][0][m*3+2]==0) break; //屏蔽空行 for (j=0;j<7;j++){ if (month[i][j][m*3+k]>0) //非正数用空格替代 cout<<setw(3)<<month[i][j][m*3+k]; //注意数组下标[m*3+k]不用+1 else cout<<" "; if (j==6&&k!=2) cout<<" | "; }//next j }//next k if (i!=5) cout<<endl; if (i==5 && (month[5][0][m*3]!=0 || month[5][0][m*3+1]!=0 || month[5][0][m*3+2]!=0)) cout<<endl; //第6行有日期则换行 }//next i } //next m cout<<endl; }//next for(;;) }//end main();
运行结果如下:
也可以将上述代码稍作修改,改为用竖排的格式输出年历,自己动手试试吧。(全文完)