信息与未来 2019

信息与未来 2019

题单

1.幸运素数

传送门

题目描述

素数,又称质数,是指除 1 1 1 和其自身之外,没有其他约数的正整数。例如 2 , 3 , 5 , 13 2,3,5,13 2,3,5,13 都是素数,而 4 , 9 , 12 , 18 4,9,12,18 4,9,12,18 则不是。特别地,规定 1 1 1 不是素数(因此自然数的质因数分解就是唯一的)。

如果一个数本身是素数,并且把最低位删除后得到的数仍是素数、再把最低位删除后得到的数仍是素数……如此往复,直到得到一个一位素数,我们就称它是“幸运素数”。以 233 233 233 为例:

  • 233 233 233 本身是素数;
  • 23 = ⌊ 233 10 ⌋ 23=\lfloor\dfrac{233}{10}\rfloor 23=10233 是素数;
  • 2 = ⌊ 23 10 ⌋ 2=\lfloor\dfrac{23}{10}\rfloor 2=1023 是素数。

因此 233 233 233 是“幸运”素数。而 211 211 211 则不是幸运素数:虽然 211 211 211 是素数,但 21 21 21 不是素数。请编程求出一定范围内的所有幸运数字。

输入格式

输入一行两个用空格分隔的正整数 m , n m,n m,n,表示我们希望求幸运素数的范围。

输出格式

从小到大输出 m , m + 1 , m + 2 , ⋯   , n m,m+1,m+2,\cdots,n m,m+1,m+2,,n 中所有的幸运素数,每行输出一个。

样例 #1
样例输入 #1
6 30
样例输出 #1
7
23
29
提示

对于 100 % 100\% 100% 的测试数据,有 1 ≤ m ≤ n ≤ 9999 1\le m\le n\le 9999 1mn9999

本题原始满分为 15 pts 15\text{pts} 15pts

分析:
  1. 判断质数 isP
  2. 不断删掉最低位,直到得到只剩一位,判断是否幸运素数 check
  3. 枚举 m 到 n 的数字,判断是否幸运素数
复杂度:≈ O ( m − n ) O(m - n) O(mn)
代码实现:
#include <bits/stdc++.h>
using namespace std;

bool isP(int n)
{
    if(n < 2)
        return false;
    for(int i = 2; i * i <= n; i++)
    {
        if(n % i == 0)
            return false;
    }
    return true;
}
bool check(int x)
{
    while(x)
    {
        if(!isP(x))
            return false;
        x /= 10;
    }
    return true;
}

int main()
{
    int m, n;
    cin >> m >> n;
    for(int i = m; i <= n; i++)
    {
        if(check(i))
            cout << i << "\n";
    }
    return 0;
}

2.粉刷矩形

传送门

题目描述

在一个 n × m n\times m n×m 的矩形上进行粉刷工作(初始时,矩形的格子无色),每次选择一个起点和一个方向,将连续的格子刷成同样的颜色,直到碰到矩形边界后停止;已经有颜色的格子中的颜色将会被覆盖,例如下图展示了依次在 ( 1 , 2 ) , ( 2 , 3 ) , ( 2 , 2 ) , ( 3 , 5 ) (1,2),(2,3),(2,2),(3,5) (1,2),(2,3),(2,2),(3,5) 格子进行向右、向上、向下、向左粉刷后每个格子的颜色。

请你编程模拟粉刷矩形的过程,并输出每个格子的颜色。

输入格式

输入第一行三个整数 n , m , k n,m,k n,m,k 表示矩形的大小是 n × m n\times m n×m、共有 k k k 次粉刷操作。
接下来 k k k 行,每行都是空格分隔的两个整数和两个字符,分别表示粉刷起点的行号、列号、粉刷的颜色(用小写字母表示)、粉刷的方向( R, U, L, D \text{R, U, L, D} R, U, L, D 分别表示向右、向上、向左、向下)。

输出格式

输出 n n n 行,每行 m m m 个字符(字符之间无空格),表示粉刷后每个格子的颜色。无色的格子输出 .

样例 #1
样例输入 #1
4 5 4
1 2 a R
2 3 b U
2 2 x D
3 5 t L
样例输出 #1
.abaa
.xb..
ttttt
.x...
提示

对于 100 % 100\% 100% 的测试数据,有 1 ≤ n , m , k ≤ 50 1\le n,m,k\le50 1n,m,k50

本题原始满分为 15 pts 15\text{pts} 15pts

分析:
  1. 初始化 smap 数组
  2. 每一次 solve 一下,模拟粉刷过程
  3. 输出 smap 数组
复杂度: O ( k n m ) O(knm) O(knm)
代码实现:

#include <bits/stdc++.h>
using namespace std;

int n, m, k;
char smap[55][55];
int a, b;
char c, d;
namespace problem{
    void init()
    {
        for (int i = 1; i <= n; i++)
            for (int j = 1; j <= m; j++)
                smap[i][j] = '.';
    }
    void solve()
    {
        cin >> a >> b >> c >> d;
        if (d == 'R')
        { // 右
            for (int i = b; i <= m; i++)
                smap[a][i] = c;
        }
        else if (d == 'U')
        { // 上
            for (int i = a; i >= 1; i--)
                smap[i][b] = c;
        }
        else if (d == 'L')
        { // 左
            for (int i = b; i >= 1; i--)
                smap[a][i] = c;
        }
        else if (d == 'D')
        { // 下
            for (int i = a; i <= n; i++)
                smap[i][b] = c;
        }
    }
};

int main()
{

    cin >> n >> m >> k;

    problem ::init();

    for (int i = 1; i <= k; i++)
    {
        problem ::solve();
    }
    for (int i = 1; i <= n; i++)
    {
        for (int j = 1; j <= m; j++)
            cout << smap[i][j];
        cout << endl;
    }
    return 0;
}

3.新斐波那契数列

传送门

题目描述

给定正整数 a ( a ≥ 1 ) a(a\ge1) a(a1),新斐波那契数列 f a f_a fa 按如下方式定义:

  • f a ( 1 ) = 1 f_a(1) = 1 fa(1)=1
  • f a ( 2 ) = a f_a(2) = a fa(2)=a
  • f a ( n ) = f a ( n − 1 ) + f a ( n − 2 )   ( n > 2 ) f_a(n) = f_a(n − 1) + f_a(n − 2)\ (n > 2) fa(n)=fa(n1)+fa(n2) (n>2)

例如,给定 a = 4 a = 4 a=4,有 f 4 ( 1 ) = 1 , f 4 ( 2 ) = 4 , f 4 ( 3 ) = 5 , f 4 ( 4 ) = 9 , f 4 ( 5 ) = 14 , ⋯ f_4(1) = 1, f_4(2) = 4, f_4(3) = 5, f_4(4) = 9, f_4(5) = 14, \cdots f4(1)=1,f4(2)=4,f4(3)=5,f4(4)=9,f4(5)=14, 现在已知新斐波那契数列中的一项 x x x,但并不知道 n n n a a a 的值是多少。请你求出所有可能的 n , a ( n ≥ 2 ) n,a(n\ge2) n,a(n2) 满足 f a ( n ) = x f_a(n) = x fa(n)=x

输入格式

你需要在一个测试数据中处理多个新斐波那契数列问题。输入第一行 T T T 表示问题的数量。

接下来 T T T 行, 每行一个整数:待求解的 x x x

输出格式

对于每个新斐波那契数列问题,按照 n n n 从小到大的顺序,输出所有可能的 n , a n,a n,a 满足 f a ( n ) = x f_a(n) = x fa(n)=x。每行输出一对 n n n a a a,由一个空格分隔。

样例 #1
样例输入 #1
2
9
123
样例输出 #1
2 9
3 8
4 4
2 123
3 122
4 61
6 24
10 3
提示

对于 60 % 60\% 60% 的测试数据,有 1 ≤ x ≤ 1 0 6 1\le x\le10^6 1x106

对于 100 % 100\% 100% 的测试数据,有 2 ≤ x ≤ 1 0 9 , 1 ≤ T ≤ 20 2\le x\le10^9,1\le T\le20 2x109,1T20

本题原始满分为 15 pts 15\text{pts} 15pts

分析:
  1. init 构造斐波那契数列
  2. 循环来依次检查数组 d 中的值是否小于等于 x,如果大于 x 就退出循环。
  3. 若 x - d[i - 2]是 d[i - 1]的倍数,则输出 i 和(x - d[i - 2]) / d[i - 1]
复杂度: O ( T log ⁡ 2 M ) O(T\log_2M) O(Tlog2M)
代码实现:

#include <bits/stdc++.h>
const int N = 1010, M = 1e9;
long long d[N] = {0, 1, 1};
namespace problem
{
    void init()
    {
        for (int i = 3;; i++)
        {
            d[i] = d[i - 1] + d[i - 2];
            if (d[i] >= M)
                break;
        }
    } // ok

    void solve()
    {
        int x;
        std ::cin >> x;
        printf("2 %d\n", x);
        for (int i = 3;; i++)
        {
            if(d[i] > x)
                break;
            if ((x - d[i - 2]) % d[i - 1] == 0)//可以构成
            {
                printf("%d %d\n", i, (x - d[i - 2]) / d[i - 1]);
            }
        }
    }
};
int main()
{
    problem ::init();
    int T;
    std ::cin >> T;
    while (T--)
    {
        problem ::solve();
    }
    return 0;
}

第四题写炸了,先看第五题

5.鸡兔同笼

传送门

题目描述

有三种动物被关在笼子里,它们分别是:

  • 鸡,每只鸡有一个头、两只脚;
  • 三脚猫,每只三脚猫有一个头,三只脚;
  • 兔子,每只兔子有一个头,四只脚。

现在知道笼子中一共有 x x x 个头、 y y y 只脚,但这个信息并不足以推导出鸡、兔和三脚猫的准确数量。这时候就需要编程来帮忙了——请你求出笼子里分别至少、至多有几只兔子。

输入格式

你需要在一个测试数据中处理多个鸡兔同笼问题。输入第一行 T T T 表示问题的数量。

接下来 T T T 行,每行两个用空格分隔的正整数 x , y x,y x,y,表示头和脚的数量。

输出格式

对于每个问题,输出一行两个空格分开的整数,表示笼子里至少和至多有几只兔子。输入数据保证每个问题至少有一种鸡、三脚猫和兔子的合法组合。

样例 #1
样例输入 #1
4
6 12
6 21
6 24
12345 40000
样例输出 #1
0 0
3 4
6 6
2965 7655
提示

对于 50 % 50\% 50% 的测试数据,有 1 ≤ x ≤ 1 0 6 1\le x\le 10^6 1x106

对于 100 % 100\% 100% 的测试数据,有 1 ≤ x ≤ 1 0 9 , 1 ≤ T ≤ 20 1\le x\le10^9,1\le T\le20 1x109,1T20

本题原始满分为 20 pts 20\text{pts} 20pts

分析:

设鸡有 a 只,三脚猫有 b 只,兔子有 c 只,则有如下关系:

a + b + c = x a+b+c=x a+b+c=x

a + 3 b + 4 c = y a+3b+4c=y a+3b+4c=y

可以解出:

$ y - 3x <= c <= (y - 2x) / 2$

这样就非常简单了

复杂度: O ( T ) O(T) O(T)
代码实现:

#include<bits/stdc++.h>
using namespace std;
int main()
{
    int T, x, y;
    cin >> T;
    while(T--)
    {
        cin >> x >> y;
        int a = y - 3 * x, b = (y - 2 * x) / 2;
        printf("%d %d\n", a >= 0 ? a : 0, b);
    }
}

6.方格覆盖

传送门

题目描述

给定一个 n × n n\times n n×n 的矩形,其中从左上角开始,对角线上连续的 k k k 个格子中有障碍物。你可以把若干 1 × 2 1\times2 1×2 的小矩形放置到该大矩形中,要求是放置的两个小矩形不能占据相同的格子,且不能碰到障碍物。例如下图是 n = 4 , k = 2 n=4,k=2 n=4,k=2 的例子,我们放置了 6 6 6 1 × 2 1\times2 1×2 的小矩形。

给定 n , k n,k n,k,请你输出一个方案,使得放置的 1 × 2 1\times2 1×2 小矩形尽可能多。可以证明, n = 4 , k = 2 n=4,k=2 n=4,k=2 时,至多只能放置 6 6 6 个小矩形。

输入格式

输入一行两个用空格分隔的正整数 n , k n,k n,k,表示矩形的大小和障碍物的数量。

输出格式

输出 n n n 行,每行 n n n 个整数(用任意数量的空格分隔)。放置的小矩形分别用 1 , 2 , ⋯ 1,2,\cdots 1,2, 编号。不放置小矩形的格子输出 0 0 0。如有多种最优方案(放置最多数量的小矩形),输出任意一种即可。

样例 #1
样例输入 #1
4 2
样例输出 #1
0 0 1 2
3 0 1 2
3 4 4 0
5 5 6 6
样例 #2
样例输入 #2
5 3
样例输出 #2
0 8 8 9 10
1 0 0 9 10
1 3 0 0 7
2 3 5 5 7
2 4 4 6 6
提示

对于 50 % 50\% 50% 的测试数据,有 1 ≤ k ≤ n ≤ 10 1\le k\le n\le10 1kn10

对于 100 % 100\% 100% 的测试数据,有 1 ≤ k ≤ n ≤ 50 1\le k\le n\le50 1kn50

本题原始满分为 20 pts 20\text{pts} 20pts

分析:

贪心:先看能不能竖着放,然后看能不能横着放

复杂度: O ( n 2 ) O(n^2) O(n2)
代码实现:

/*
*/

#include <bits/stdc++.h>
using namespace std;

int a[100][110];

int main()
{
    int n, k, ttt = 0;
    cin >> n >> k;
    for (int i = 1; i <= k; i++)
        a[i][i] = -114514;            // 障碍
    for (int i = 1; i <= n; i++) // 行
        for (int j = 1; j < n; j++)
            if (!(a[i][j] || a[i][j + 1]))
            {
                a[i][j] = a[i][j + 1] = ++ttt;
            }

    for (int j = 1; j <= n; j++) // 列
        for (int i = 1; i < n; i++)
        {
            if (!(a[i][j] || a[i + 1][j]))
            {
                a[i][j] = a[i + 1][j] = ++ttt;
            }
        }

    for (int i = 1; i <= n; i++)
    {
        for (int j = 1; j <= n; j++)
        {
            if (a[i][j] == -114514)
            {
                printf("0 ");
            }
            else
            {
                printf("%d ", a[i][j]);
            }
        }
        puts("");
    }
    return 0;
}

  • 23
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值