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

    xiaoxiao2025-09-03  337

    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.先求底边字符数,进而求每行空格字符数

    转载请注明原文地址: https://ju.6miu.com/read-1302274.html
    最新回复(0)