PTA 520钻石争霸赛题解

第一题

7-1 考试周 (5分)

在这里插入图片描述
考试周快到了,浙江大学的电子屏又调皮了…… 本题请你帮小编写一个自动倒计时的程序,对给定的日期(例如“腊八”就对应 8)和倒计时天数(例如电子屏上的“四天之后”就对应 4),自动调整公式里的分母(例如 8/2=4 里面的那个 2)。

输入格式:

输入在一行中给出两个正整数:A 是给定的日期,不超过 30;B 是倒计时天数,不超过 10。

输出格式:

在一行中输出公式 A/X=B,其中 X 是满足等式的数字,输出时保留小数点后 1 位即可。

输入样例:

8 3

输出样例:

8/2.7=3

思路:

就是将 A / B = X A/B=X A/B=X A / X = B A/X=B A/X=B输出。

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <deque>
#include <stack>
#include <cctype>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(int x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
int main() {
    double n, m;
    cin >> n >> m;
    printf("%.0lf/%.1lf=%.0lf", n, n / m, m);
}

第二题

7-2 真的恭喜你 (10分)

当别人告诉你自己考了 x 分的时候,你要回答说:“恭喜你考了 x 分!”比如小明告诉你他考了90分,你就用汉语拼音打出来 gong xi ni kao le 90 fen!。

但是如果小明没考好,比如只考了 20 分,你也“恭喜”人家就不对了。这时候你应该安慰他说:“考了 20 分别泄气!”用汉语拼音写出来就是 kao le 20 fen bie xie qi!。

输入格式:

输入在一行里给出一位小朋友的分数。这个分数是一个 0 到 100 之间的整数。

输出格式:

在一行中输出你对这位小朋友说的话。如果人家考到不低于 90 分,就说 gong xi ni kao le X fen!;如果不到 90 分,就说 kao le X fen bie xie qi!。其中 X 是小朋友输入的分数。

输入样例 1:

95

输出样例 1:

gong xi ni kao le 95 fen!

输入样例 2:

89

输出样例 2:

kao le 89 fen bie xie qi!

思路:

i f if if判断输入是否大于等于 90 90 90

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <deque>
#include <stack>
#include <cctype>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(int x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
int main() {
    int n;
    cin >> n;
    if (n >= 90) printf("gong xi ni kao le %d fen!", n);
    else printf("kao le %d fen bie xie qi!", n);
}

第三题:

7-3 平均成绩 (10分)

给定一所学校学生的体育成绩,请你统计全班的平均成绩和男生、女生的平均成绩。

输入格式:

输入首先在第一行中给出一个正整数 N(≤10000),即全校学生人数。最后 N 行,每行按照格式 性别 成绩 给出一位学生的信息。其中 性别 为 1 表示男生,0 表示女生;成绩 是一个 0 到 100 之间的整数。

输出格式:

在一行中依次输出全班的平均成绩、男生的平均成绩、女生的平均成绩。输出小数点后1位,数字间有一个空格分隔。

注意:如果全是男生或全是女生,则缺少的性别就没有办法计算平均分,相应的位置应该输出一个 X 表示没有。

输入样例 1:

5
1 97
0 91
0 98
1 95
1 90

输出样例 1:

94.2 94.0 94.5

输入样例 2:

5
0 97
0 91
0 98
0 95
0 90

输出样例 2:

94.2 X 94.2

思路:

用一个二维的 v e c t o r vector vector存储这些人数男存 1 1 1,女存 2 2 2,如果 s i z e = 0 size=0 size=0的情况,就说明此时没有人,那么输出 X X X,否则计算 s u m / s i z e sum/size sum/size

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <deque>
#include <stack>
#include <cctype>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(int x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
veci v[5];
int main() {
    int n, sum = 0;
    cin >> n;
    for (int i = 0; i < n; i++) {
        int a, b;
        cin >> a >> b;
        v[a].push_back(b);
        sum += b;
    }
    printf("%.1lf ", sum * 1.0 / n);

    sum = 0;
    for (int i = 0; i < v[1].size(); i++) {
        sum += v[1][i];
    }
    if (v[1].size() == 0) cout << "X ";
    else printf("%.1lf ", sum * 1.0 / v[1].size());

    sum = 0;
    for (int i = 0; i < v[0].size(); i++) {
        sum += v[0][i];
    }
    if (v[0].size() == 0) cout << "X";
    else printf("%.1lf", sum * 1.0 / v[0].size());

    return 0;
}

第四题

7-4 古风A+B (15分)

给定两个绝对值不超过 109的整数 A 和 B,计算它们的和 A+B 是非常简单的事,但要将和这个数字按照中国古风从上到下竖着输出,就不是那么简单了吧?

输入格式:

输入在一行中给出 2 个绝对值不超过 109的整数 A 和 B,其间以空格分隔。

输出格式:

将 A+B 数值的每一位从上到下竖着输出。如果是负数,那么负号占第 1 行,其它数字不用输出符号。

输入样例:

-6666 233

输出样例:

− -
6 4 3 3 6\\4\\3\\3 6433

思路:

先将 a + b a+b a+b,然后可以用递归输出每一位,如果这个是负数的话,那么负号要提前输出。

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <deque>
#include <stack>
#include <cctype>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(int x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
void print(int x) {
    if (x == 0) return ;
    print(x / 10);
    cout << x % 10 << endl;
}
int main() {
    int a, b;
    cin >> a >> b;
    int x = a + b;
    if (x == 0) {
        cout << "0" << endl;
        return 0;
    }
    if (x < 0) x = -x, cout << "-\n";
    print(x);
}

第五题

7-5 猜近似数字 (15分)

甲想好了一个 n 位数字让乙来猜,只要猜的数字位数正确,至多有 1 位与谜底不同,且不同的数字相差不超过 1,就算猜对了。例如谜底是 67,若乙猜 66、68、77、57,就都算对了;猜 167 就不能算对。

输入格式:

输入首先在第一行给出一个不超过 1000 位数的、最高位不是 0 的正整数,是甲给出的谜底。随后若干行,每行给出一个乙猜的数字,都是最高位不是 0 的正整数。直到出现 -1 表示输入结束,这个数字不要做任何处理。题目保证乙至少猜了一次。

输出格式:

对每一个乙猜的数字,如果猜对了就输出 Yes,否则输出 No。

输入样例:

12345678909876
2345678909876
12345678900876
12345678809876
12345678909888
1234567890987
-1

输出样例:

No
No
Yes
No
No

思路:

将全部遍历一遍,首先要保证遍历的时候只有一个是不同的,然后再进行判断这个不同的数字是否相差 1 1 1

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <deque>
#include <stack>
#include <cctype>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(int x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
string s, t;
int main() {
    cin >> s;
    while (cin >> t && t != "-1") {
        if (t.size() != s.size()) {
            cout << "No\n";
            continue;
        }
        int cnt = 0;
        for (int i = 0; i < s.size(); i++) {
            if (s[i] != t[i]) {
                cnt++;
                if (s[i] != t[i] - 1 && s[i] != t[i] + 1) cnt++;
            }
        }
        if (cnt <= 1) {
            cout << "Yes\n";
        } else cout << "No\n";
    }
    return 0;
}

第六题

7-6 随机输一次 (20分)

大家应该都会玩“锤子剪刀布”的游戏:两人同时给出手势,胜负规则如图所示:

FigCJB.jpg

现要求你编写一个控制赢面的程序,根据对方的出招,给出对应的赢招。但是!为了不让对方意识到你在控制结果,你需要隔 K 次输一次,其中 K 是系统设定的随机数。

输入格式:

输入首先在第一行给出正整数 N(≤10),随后给出 N 个系统产生的不超过 10 的正随机数 { K1,K2,⋯,KN},数字间以空格分隔。这意味着第 i(i=0,1,⋯,N−1)次输局之后应该隔 Ki+1次再让下一个输局。如果对方出招太多,则随机数按顺序循环使用。例如在样例中,系统产生了 3 个随机数 {2, 4, 1},则你需要:赢 2 次,输 1 次;赢 4 次,输 1 次;赢 1 次,输 1 次;然后再次回到第 1 个随机数,赢 2 次,输 1 次。

之后每行给出对方的一次出招:“ChuiZi”代表“锤子”、“JianDao”代表“剪刀”、“Bu”代表“布”。“End”代表输入结束,这一行不要作为出招处理。输入保证对方至少出了一招。

输出格式:

对每一个输入的出招,按要求输出赢或输局的招式。每招占一行。

输入样例:

3 2 4 1
ChuiZi
JianDao
Bu
JianDao
Bu
ChuiZi
ChuiZi
ChuiZi
JianDao
Bu
JianDao
Bu
ChuiZi
End

输出样例:

Bu
ChuiZi
ChuiZi
ChuiZi
JianDao
Bu
Bu
JianDao
ChuiZi
ChuiZi
ChuiZi
JianDao
JianDao

思路:

比赛是 a i − w i n − > 1 − l o s e a_i-win->1-lose aiwin>1lose这样的,那么我们可以设置一个 f l a g flag flag,当 f l a g = a i flag=a_i flag=ai的时候就说明此时到达了输的时候,因为 f l a g flag flag 0 0 0开始,然后将 f l a g = 0 , c n t flag=0,cnt flag=0,cnt++, c n t cnt cnt指的就是 i i i,但是 c n t cnt cnt可能会超过 k k k,所以要对 c n t cnt cnt取模。

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <deque>
#include <stack>
#include <cctype>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(int x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
int a[20] = {0};
string s;
void printwin(string x) {
    if (x == "ChuiZi") cout << "Bu\n";
    else if (x == "JianDao") cout << "ChuiZi\n";
    else cout << "JianDao\n";
}
void printlose(string x) {
    if (x == "ChuiZi") cout << "JianDao\n";
    else if (x == "JianDao") cout << "Bu\n";
    else cout << "ChuiZi\n";
}
int main() {
    int k;
    read(k);
    for (int i = 0; i < k; i++) read(a[i]);
    int cnt = 0, flag = 0;
    while (cin >> s && s != "End") {
        if (flag == a[cnt % k]) {
            printlose(s);
            flag = 0;
            cnt++;
        } else {
            printwin(s);
            flag++;
        }
    }
    return 0;
}

第七题

7-7 阶乘的非零尾数 (20分)

“求 N 阶乘末尾的第一个非零数字”是一道常见的企业笔试题。这里我们略微做个变化,求 N 阶乘末尾的第一个非零 K 位数,同时输出末尾有多少个零。

输入格式:

输入给出一个不超过 107的正整数 N 和要求输出的位数 0<K<10。

输出格式:

在一行中输出 N 阶乘末尾的第一个非零 K 位数(注意前导零也要输出)、以及末尾 0 的个数,其间以 1 个空格分隔。

输入样例:

18 5

输出样例:

05728 3

思路:

我们可以先将阶乘里面的 2 , 5 2,5 2,5都取出来,因为只有 2 , 5 2,5 2,5才可以得出 10 10 10,然后这样取余 1 0 k 10^k 10k就对答案没有影响了,但是最后的时候,对于之前取出的 2 , 5 2,5 2,5可能不能完全用完,那么我们就将多余的 2 2 2 5 5 5再进行相乘取余,最后得出的就是答案了。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(int x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
int main() {
    ll n, k, x = 1, y = 1;
    read(n), read(k);
    for (int i = 1; i <= n; i++) y *= i;
    for (int i = 0; i < k; i++) x *= 1ll * 10;
    ll num = 1, cnt = 0;
    bool flag = true;
    ll cnt2 = 0, cnt5 = 0;
    for (int i = 1; i <= n; i++) {
        ll mm = i;
        while (mm % 5 == 0) mm /= 5, cnt5++;
        while (mm % 2 == 0) mm /= 2, cnt2++;
        num *= mm;
        num %= x;
    }
    cnt = min(cnt5, cnt2);
    for (int i = 0; i < cnt5 - cnt; i++) num = 5ll * num % x;
    for (int i = 0; i < cnt2 - cnt; i++) num = 2ll * num % x;
    ll tmp = num, cntt = 0;
    while (num) {
        num /= 10;
        cntt++;
    }
    //if (flag == false) {
        for (int i = 0; i < k - cntt; i++) cout << "0";
    //}
    cout << tmp << " " << cnt << endl;
    return 0;
}

第八题

7-8 三足鼎立 (25分)

当三个国家中的任何两国实力之和都大于第三国的时候,这三个国家互相结盟就呈“三足鼎立”之势,这种状态是最稳定的。

现已知本国的实力值,又给出 n 个其他国家的实力值。我们需要从这 n 个国家中找 2 个结盟,以成三足鼎立。有多少种选择呢?

输入格式:

输入首先在第一行给出 2 个正整数 n(2≤n≤105)和 P(≤109),分别为其他国家的个数、以及本国的实力值。随后一行给出 n 个正整数,表示n 个其他国家的实力值。每个数值不超过 109,数字间以空格分隔。

输出格式:

在一行中输出本国结盟选择的个数。

输入样例:

7 30
42 16 2 51 92 27 35

输出样例:

9

样例解释:

能联合的另外 2 个国家的 9 种选择分别为:

{16, 27}, {16, 35}, {16, 42}, {27, 35}, {27, 42}, {27, 51}, {35, 42}, {35, 51}, {42, 51}。

思路:

如果直接暴力的话,肯定是超时了,所以我们可以用二分查找,查找对于 a i a_i ai来说可以满足的条件的情况,有两种情况:

  • 第一种就是 p p p不是最大的情况,那么我们只要找到大于等于 a i + p a_i+p ai+p的数字。
  • 第二种就是 p p p是最大的情况,那么我们就要找到大于 p − a i p-a_i pai的情况。

最后我们就可以知道这个区间是 ( p − a i , a i + p ) (p-a_i,a_i+p) (pai,ai+p),只要将这个区间里面的全部存在的数字相加就是答案了,如果是用哈希的方式的话,复杂度是 O ( P ) O(P) O(P)也太大,所以只能用二分,复杂度 O ( n l o g n ) O(nlogn) O(nlogn)

#include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
#include <queue>
#include <vector>
#include <string>
#include <cmath>
#include <set>
#include <map>
#include <deque>
#include <stack>
#include <cctype>
using namespace std;
typedef long long ll;
typedef vector<int> veci;
typedef vector<ll> vecl;
typedef pair<int, int> pii;
template <class T>
inline void read(T &ret) {
    char c;
    int sgn;
    if (c = getchar(), c == EOF) return ;
    while (c != '-' && (c < '0' || c > '9')) c = getchar();
    sgn = (c == '-') ? -1:1;
    ret = (c == '-') ? 0:(c - '0');
    while (c = getchar(), c >= '0' && c <= '9') ret = ret * 10 + (c - '0');
    ret *= sgn;
    return ;
}
inline void out(ll x) {
    if (x > 9) out(x / 10);
    putchar(x % 10 + '0');
}
const int maxn = 1e5 + 10;
int a[maxn] = {0};
int main() {
    int n, p, x;
    ll ans = 0;
    read(n), read(p);
    for (int i = 0; i < n; i++) read(a[i]);
    sort(a, a + n);
    for (int i = 0; i < n - 1; i++) {
        int x = lower_bound(a + i + 1, a + n, a[i] + p) - a;
        int y = upper_bound(a + i + 1, a + n, p - a[i]) - a;
        ans += 1ll * max(0, x - y);
    }
    out(ans);
    return 0;
}

©️2020 CSDN 皮肤主题: 编程工作室 设计师:CSDN官方博客 返回首页