本题要求你写个程序把给定的符号打印成沙漏的形状。例如给定17个“*”,要求按下列格式打印
*****
***
*
***
*****
所谓“沙漏形状”,是指每行输出奇数个符号;各行符号中心对齐;相邻两行符号数差2;符号数先从大到小顺序递减到1,再从小到大顺序递增;首尾符号数相等。
给定任意N个符号,不一定能正好组成一个沙漏。要求打印出的沙漏能用掉尽可能多的符号。
输入格式:
输入在一行给出1个正整数N(≤1000)和一个符号,中间以空格分隔。
输出格式:
首先打印出由给定符号组成的最大的沙漏形状,最后在一行中输出剩下没用掉的符号数。
输入样例:
19 *
输出样例:
*****
***
*
***
*****
2
#include<iostream>
using namespace std;
int main() {
//关键在于发现规律——行数等于第一行的符号数
//沙漏是从一行的一个符号依次往上下扩展一行,同时每行增加两个符号,故第一行的符号数永远等于总行数
int num, row;
char a;
cin >> num >> a;
if (num) {
row = 1;
num -= 1;
}
else row = 0; //若输入num为0,则不打印
while (num >= 2 * (row + 2)) {
//顺序不可颠倒
num -= 2 * (row + 2);
row += 2;
}
int temp = row;
//打印上半部分,即到只有一个符号那一行
for (int j = 0; j < (row + 1) / 2; j++, temp -= 2) {
//打印空格,上半部分空格数为行数减一(注意只有左边要打空格,右边不用!!!打印完符号后直接换行即可)
for (int k = 0; k < j; k++) {
cout << ' ';
}
//打印符号,第一行符号数等于行数,之后递减2(递减2应该放在外层循环里)
for (int l = 0; l < temp; l++) {
cout << a;
}
cout << endl;
}
//打印下半部分,n为每一行的空格数
//k是打印符号的for循环的计数变量,为了让增2能在外层循环进行故将它在外层循环中定义
for (int j = 0, k = 0; j < row - (row + 1) / 2; j++, k -= 2*j) {
//打印空格,该行下面还有几行就打印几个空格
for (int m = 0; m < row - (row + 1) / 2 - j - 1; m++) {
cout << ' ';
}
//打印符号,下半部分第一行符号数为行数加2,以题目为例,第四行是下半部分的第一行,1+2=3,之后再递增2(也要置于外层循环里)
for (; k < 3; k ++) {
cout << a;
}
k = 0;//更新计数器k
cout << endl;
}
cout << num;
return 0;
}
在看了一个博客后按自己的想法修改了一部分,大体思路相似,细节方面有所差异,已经通过了
在此附上原博客链接
有什么更好的想法或者该代码有什么问题欢迎大家提出