PAT乙级.1027. 打印沙漏(20)

该博客介绍了如何使用编程打印出沙漏形状,详细讲解了思路和代码实现,包括两个版本的代码,并通过数学推导解决等差数列求和问题,以确保使用尽可能多的给定符号。文章最后分享了通过此题目的收获,如解决问题的方法和等差数列知识的应用。
摘要由CSDN通过智能技术生成

1027. 打印沙漏(20)


题目:

本题要求你写个程序把给定的符号打印成沙漏的形状。例如给定17个“*”,要求按下列格式打印

打印沙漏
所谓“沙漏形状”,是指每行输出奇数个符号;各行符号中心对齐;相邻两行符号数差2;符号数先从大到小顺序递减到1,再从小到大顺序递增;首尾符号数相等。

给定任意N个符号,不一定能正好组成一个沙漏。要求打印出的沙漏能用掉尽可能多的符号。

输入格式:

输入在一行给出1个正整数N(<=1000)和一个符号,中间以空格分隔。

输出格式:

首先打印出由给定符号组成的最大的沙漏形状,最后在一行中输出剩下没用掉的符号数。

输入样例:

19 *

输出样例:

这里写图片描述
2

PAT链接


思路:

这里写图片描述
1.首先要得到不同数字所能对应的底边图案数,方法是1+3+5+……不断累加,直到数字比n大跳出
2.依次打印上三角和下三角
3.每次打印字符时,cnt++

//得到底边的符号数
int getCol(int n)
{
  int upper = 0;
  int sum = 0;
  int i = 1;
  for (i; i<=n; i++)
  {
    upper += (i * 2 - 1);
    sum = 2 * upper - 1;
    if (sum > n) break;
  }
  return 2*i-3;
}   //测试点2为n=1的情况

2.根据底边图案数分别打印上半部分和下半部分


代码:

version1.0
/**
* @tag     PAT_B_1027
* @authors R11happy (xushuai100@126.com)
* @date    2016-8-14 5:26-6:10
* @version 1.0
* @Language C++
* @Ranking  290/2389
* @function null
*/

#include <cstdio>
#include <cstdlib>
#include <cstring>
/*
//得到底边的符号数
int getCol(int n)
{
  int upper = 0;
  int sum = 0;
  int i;
  for (i = 0; i<n; i++)
  {
    upper += (i * 2 + 1);
    sum = 2 * upper - 1;
    if (sum > n) return 2 * i - 1;
    if (sum == n)   return 2*i + 1; //考虑测试点2:n=1的情况
  }
  return -1;
}*/

//得到底边的符号数
int getCol(int n)
{
  int upper = 0;
  int sum = 0;
  int i = 1;
  for (i; i<=n; i++)
  {
    upper += (i * 2 - 1);
    sum = 2 * upper - 1;
    if (sum > n) break;
  }
  return 2*i-3;
}   //测试点2为n=1的情况

int main(int argc, char const *argv[])
{
  int i, N;
  char ch;
  int col;
  int cnt = 0;
  scanf("%d %c", &N, &ch);
  col = getCol(N);
  if (N)
  {
    //打印上半部分(包括中间的)
    for (i = col; i >= 1; i -= 2)
    {
      for (int k = 0; k<(col - i) / 2; k++)
      {
        printf(" ");
      }
      for (int j = 0; j<i; j++)
      {
        printf("%c", ch);
        cnt++;
      }
      printf("\n");
    }
    //打印下半部分
    for (i += 4; i <= col; i += 2)
    {
      for (int k = 0; k<(col - i) / 2; k++)
      {
        printf(" ");
      }
      for (int j = 0; j<i; j++)
      {
        printf("%c", ch);
        cnt++;
      }
      printf("\n");
    }
  }
  //打印剩余符号
  printf("%d\n", N - cnt);
  return 0;
}
version2.0(参考晴神代码)
/**
* @tag     PAT_B_1027
* @authors R11happy (xushuai100@126.com)
* @date    2016-8-14 5:26-6:10
* @version 1.0
* @Language C++
* @Ranking  290/2389
* @function null
*/

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>

int main(int argc, char const *argv[])
{
    int n;
    char ch;
    scanf("%d %c", &n, &ch);
    int bottom = floor(sqrt(2*(n+1))) - 1;  //得到底边的字符数
    if(bottom % 2 == 0) bottom--;   //偶数修正,得到奇数
    int used = (bottom+1)*(bottom+1)/2 - 1;
    //打印上半部分(包括中间的)
    for (int i = bottom; i >= 1; i -= 2)
    {
      for (int k = 0; k<(bottom - i) / 2; k++)
      {
        printf(" ");
      }
      for (int j = 0; j<i; j++)
      {
        printf("%c", ch);
      }
      printf("\n");
    }
    //打印下半部分
    for (int i = 3; i <= bottom; i += 2)
    {
      for (int k = 0; k<(bottom - i) / 2; k++)
      {
        printf(" ");
      }
      for (int j = 0; j<i; j++)
      {
        printf("%c", ch);
      }
      printf("\n");
    }
    printf("%d\n", n - used );
    return 0;
}


收获:

1.用数学推导解决问题:

    int bottom = floor(sqrt(2*(n+1))) - 1;  //得到底边的字符数
    if(bottom % 2 == 0) bottom--;   //偶数修正,得到奇数
    int used = (bottom+1)*(bottom+1)/2 - 1; //总输出的非空格字符数

2.等差数列求和公式
3.先求底边字符数,进而求每行空格字符数

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值