『ACM C++』PTA浙大 | 基础题 - 打印沙漏
《数据结构》开课前的一些小作业练习,可能因为一个寒假都没有打C++手生了,整个寒假都在帮拍电影做后期特效,导致这道题居然用了两个钟去AC,深感惭愧,作个标记吧,下面上题。
一首好曲推荐:同时我也设置成了我的博客背景音乐
《开启新征程》 —— 流浪地球电影的片尾曲
收藏链接:https://music.163.com/#/song?id=1343461538
------------------------------------------------题目----------------------------------------------------------
本题要求你写个程序把给定的符号打印成沙漏的形状。例如给定17个“*”,要求按下列格式打印
*****
***
*
***
*****
所谓“沙漏形状”,是指每行输出奇数个符号;各行符号中心对齐;相邻两行符号数差2;符号数先从大到小顺序递减到1,再从小到大顺序递增;首尾符号数相等。
给定任意N个符号,不一定能正好组成一个沙漏。要求打印出的沙漏能用掉尽可能多的符号。
输入格式:
输入在一行给出1个正整数N(≤1000)和一个符号,中间以空格分隔。
输出格式:
首先打印出由给定符号组成的最大的沙漏形状,最后在一行中输出剩下没用掉的符号数。
输入样例:
*
输出样例:
*****
***
*
***
*****
------------------------------------------------题目----------------------------------------------------------
(一) 题目分析:
首先经过题目分析,可知特殊符号数跟行数有着某种特殊关系,除去中间的一颗星星*,可以发现上半部分的是个递增的等差数列:3,5,7,9,.......,可得通项公式2n+1(其中n = 上半部分行数 = 下半部分行数)
那么求和即为Sn = n2+2n,这是上半部分的。那么上下两部分即可得到:
总**用量:2( n2+2n) = 2n2+4n
到这里我们就可以算出输入的数能生成几行了。
(二)代码分块:
首先,先计算得出能上下部分各能生成几行:
for(i = ;;i++)
{
temp = *i*i+*i;
if(temp > cNumber-) break;
Max = temp;
}
temp临时存放下一行所需的总量,如果这个总量超过了输入的量数,那么就break打破永真循环,不把temp值赋值给Max行数,否则就把temp的行数赋值给Max变量。
第二步,绘制上半部分:
for(temp = i-;temp>;temp--)
{
for(int count = ;count != (i-)-temp;count++) printf(" ");
for(int count = ;count<(*temp+);count++) printf("%c",symbol);
printf("\n");
}
这时候的temp就是行数了,i-1是因为永真循环的时候被i++多了一次,所以需要减掉,以样例为例,输入19后,此时temp应该为2,也就是给上半部分绘制两行。
因为题目要求中心对齐,我们同时也发现空格和行数之间的关系,例如第一行是没有空格的,第二行有一个空格,第三行有两个空格,所以通过空格数和行的关系去输出即可,然后每一行最后都换行,代码就形成了。
第三步,绘制下半部分:
for(temp = ;temp<=i-;temp++)
{
for(int count = ;count != (i-)-temp;count++) printf(" ");
for(int count = ;count<(*temp+);count++) printf("%c",symbol);
printf("\n");
}
此时需要注意,最外层的for,在绘制上半部分的时候,是temp>0,是不取边界的,如果取了边界会使上部分多画一行单个*。我这里选择的是下半部分取绘制这单个*,因此需要temp<=i-1,取到边界。
(三)AC代码:
#include<iostream>
#include<stdio.h>
using namespace std;
int cNumber,Max,i,temp;
char symbol;
int main()
{
scanf("%d %c",&cNumber,&symbol);
for(i = ;;i++)
{
temp = *i*i+*i;
if(temp > cNumber-) break;
Max = temp;
}
for(temp = i-;temp>;temp--)
{
for(int count = ;count != (i-)-temp;count++) printf(" ");
for(int count = ;count<(*temp+);count++) printf("%c",symbol);
if(temp != ) printf("\n");
}
for(temp = ;temp<=i-;temp++)
{
for(int count = ;count != (i-)-temp;count++) printf(" ");
for(int count = ;count<(*temp+);count++) printf("%c",symbol);
printf("\n");
}
printf("%d",cNumber-Max-);
return ;
}
(四)AC截图:

(五)解后分析
题目难度不大,刚开始可能手生,在写for的时候总是渲染多一行或者找错最大行数。
推荐知识:
1、将声明变量放在main主函数外,可以申明稍微大型一点的变量,因为main的函数栈容量有限。同时还有一个好处,在函数外申明数组的时候会自动默认全为0清零。
2、推荐使用scanf("%d %c",&cNumber,&symbol); printf("%c",symbol); 这种C语言输入输出方式,在ACM中将能大大减少输入输出所占用的算法时间。
(六)优质解法更新:
这是在写博客的时候突然想到可以从上到下一次性绘图,写完博客就去试了一下,还真的可以,空间复杂度大大降低了,同样也AC,现在贴上代码:
#include<iostream>
#include<cmath>
#include<stdio.h>
using namespace std;
int cNumber,Max,i,temp;
char symbol;
int main()
{
scanf("%d %c",&cNumber,&symbol);
for(i = ;;i++)
{
temp = *i*i+*i;
if(temp > cNumber-) break;
Max = temp;
}
for(temp = *(i-);temp>=;temp--)
{
for(int count = i-;count != abs((i-)-temp);count--) printf(" ");
for(int count = ;count<*abs(temp-i+) + ;count++) printf("%c",symbol);
printf("\n");
}
printf("%d",cNumber-Max-);
return ;
}

可以发现内存使用量大大减少,将两块的上下绘制合并到一块之后代码空间复杂度下降很多。
注:如果有更好的解法,真心希望您能够评论留言贴上您的代码呢~
『ACM C++』PTA浙大 | 基础题 - 打印沙漏的更多相关文章
- 『ACM C++』PTA浙大 | 基础题 - Have Fun with Numbers
连着这两道都是开学前数构老师的“爱心作业”,还没上课开学就给我们布置作业了,这道题有点小坑,也经常遇到类似的问题,特地拿出来记录一下. -------------------------------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 021-024
忙疯警告,这两天可能进度很慢,下午打了一下午训练赛,训练赛的题我就不拿过来的,pta就做了一点点,明天又是满课的一天,所以进度很慢啦~ -------------------------------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 007-011
真的是忙头晕了,学业.ACM打题.班级活动.自学新东西,哇这充实的大学~ ------------------------------------------------L1-007--------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 036-037
这几天比较忙,所以随便做做水题了,得赶紧把英剧搞完啊啊啊啊啊啊 ------------------------------------------------L1-036-------------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 001-006
应师兄要求,在打三月底天梯赛之前要把PTA上面的练习集刷完,所以后面的时间就献给PTA啦~ 后面每天刷的题都会把答案代码贡献出来,如果有好的思路想法也会分享一下~ 欢迎大佬提供更好的高效率算法鸭~ - ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 029-033
哈哈,今天开始我也是学车人了~ 开始一千多道疯狂刷题~ ------------------------------------------------L1-029------------------ ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 025-026
满课一天,做25的时候还疯狂WA,进度可以说是很慢了 哭泣 ------------------------------------------------L1-025---------------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 018-020
终于一周有这么一天能够安静下来好好学习打打题,还是很美滋滋的哈哈~加油加油~ ------------------------------------------------L1-018------- ...
- 『ACM C++』 PTA 天梯赛练习集L1 | 052-053
今日刷题,水题水题 ------------------------------------------------L1-052------------------------------------ ...
随机推荐
- WebGrease—异常来自 HRESULT:0x80131040
一.错误源: 未能加载文件或程序集“WebGrease, Version=1.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35”或它的某一 ...
- urllib模块的使用
Python3学习笔记(urllib模块的使用) 1.基本方法 urllib.request.urlopen(url, data=None, [timeout, ]*, cafile=None, ca ...
- VS如何自动创建函数头 函数说明
这样创建,这种方法在VS2008和VS2013下都可以. 这是效果 如何自定义格式呢? 第一步点击这个按钮 这个工具条需要安装VAssistX,如何安装这里不细说了. 第二步,点击这个按钮 第三步 点 ...
- android学习之VelocityTracker
今天大概的学习了一下 VelocityTracker ,用来监控手势速度的,在View移动过程中,可以有个缓冲的移动. 我觉得初步的使用起来是很简单的. 首先移动依靠的是Scroller类,Scro ...
- java 理解有符号数和无符号数
转至:http://jinguo.iteye.com/blog/212049 理解有符号数和无符号数负数在计算机中如何表示呢? 这一点,你可能听过两种不同的回答. 一种是教科书,它会告诉你:计算机用“ ...
- CSS3中的Flexbox弹性布局(一)
CSS3引入了一种新的布局模式——Flexbox布局,即伸缩布局盒模型(Flexible Box),用来提供一个更加有效的方式制定.调整和分布一个容器里项目布局,即使它们的大小是未知或者动态的,这里简 ...
- Hive的UDF(用户自定义函数)开发
当 Hive 提供的内置函数无法满足你的业务处理需要时,此时就可以考虑使用用户自定义函数(UDF:user-defined function). 测试各种内置函数的快捷方法: 创建一个 dual 表 ...
- windows下使用VNC进行远程连接
在 windows 电脑上安装 VNC,包含 VNC server 和 VNC viewer,如果仅需要被操控或操控他人,选择型下载安装 VNC server 或 VNC viewer 即可. 在需要 ...
- Perl实用中文处理步骤(修改版)
发信人: FenRagwort (泽), 信区: Perl标 题: Perl实用中文处理步骤(修改版)发信站: 水木社区 (Mon Feb 14 12:52:14 2011), 转信 (修改版 感谢 ...
- python入门20 导入模块(引包)
1 引包: import module 或 import module.module1 或 from module import module1,module2...等 2 import xx ...