C++数组在年历打印中的运用

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();

运行结果如下:

也可以将上述代码稍作修改,改为用竖排的格式输出年历,自己动手试试吧。(全文完)

  • 17
    点赞
  • 42
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Hann Yang

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值