第十二届蓝桥杯C/C++ 大学B组 第一场

  • A.空间

题目描述
小蓝准备用 256MB 的内存空间开一个数组,数组的每个元素都是 32 位
二进制整数,如果不考虑程序占用的空间和维护内存需要的辅助空间,请问
256MB 的空间可以存储多少个 32 位二进制整数?

问题分析
1 byte = 8 bit , 32 b = 4 byte
1 MB = 1024 KB = 1024 * 1024 B

#include<bits/stdc++.h>
using namespace std;
int main()
{
	cout<<256*1024*1024/4<<endl;
	return 0;
}

 

 

  • B.卡片

题目描述
小蓝有很多数字卡片,每张卡片上都是数字 0 到 9。
小蓝准备用这些卡片来拼一些数,他想从 1 开始拼出正整数,每拼一个,
就保存起来,卡片就不能用来拼其它数了。
小蓝想知道自己能从 1 拼到多少。
例如,当小蓝有 30 张卡片,其中 0 到 9 各 3 张,则小蓝可以拼出 1 到 10,
但是拼 11 时卡片 1 已经只有一张了,不够拼出 11。
现在小蓝手里有 0 到 9 的卡片各 2021 张,共 20210 张,请问小蓝可以从 1
拼到多少?

问题分析
模拟


#include <iostream>

using namespace std;

int s[10];

bool check(int x)
{
    while (x)
    {
        int t = x % 10;
        x /= 10;
        if ( -- s[t] < 0) return false; 
    }
    return true;
}

int main()
{
    for (int i = 0; i < 10; i ++ ) s[i] = 2021;

    for (int i = 1; ; i ++ )
        if (!check(i))
        {
            cout << i - 1 << endl;
            return 0;
        }
    return 0;
}//答案:3181
 /*检验
 #include<bits/stdc++.h>
 using namespace std;
 int main()
 {
 int n=0;
 for(int i=1;i<=3181;i++)
 {
 int m=i;
 while(m)
 {
 if(m%10==1)//1最先被消耗完
 {
 n++;
}
m/=10;
}
}
cout<<n<<endl; //n=2021得证
return 0; 
}*/
作者:李想念

答案:3181

  • C:直线

【问题描述】
在平面直角坐标系中,两点可以确定一条直线。如果有多点在一条直线上,
那么这些点中任意两点确定的直线是同一条。
给定平面上 2 × 3 个整点 {(x, y)|0 ≤ x < 2, 0 ≤ y < 3, x ∈ Z, y ∈ Z},即横坐标
是 0 到 1 (包含 0 和 1) 之间的整数、纵坐标是 0 到 2 (包含 0 和 2) 之间的整数
的点。这些点一共确定了 11 条不同的直线。
给定平面上 20 × 21 个整点 {(x, y)|0 ≤ x < 20, 0 ≤ y < 21, x ∈ Z, y ∈ Z},即横
坐标是 0 到 19 (包含 0 和 19) 之间的整数、纵坐标是 0 到 20 (包含 0 和 20) 之
间的整数的点。请问这些点一共确定了多少条不同的直线。

分析:暴力枚举任意两个点的坐标,然后记录他们的斜率
和截距放到set里面去重
已知:
二元一次方程组:y=kx+b;
y1=kx1+b ——推出  k=y2-y1/x2-x1
y2=kx2+b         b=x2y1-x1y2/x2-x1

#include <iostream>
#include <algorithm>
#include <cmath>

using namespace std;

const int N = 200000; // 大概四百个点,每两个点构成一条直线,大概160000条 

int n;
struct Line
{
    double k, b; // 每条直线可以用一对斜率和截距表示 
    bool operator< (const Line& t) const
    {
        if (k != t.k) return k < t.k;
        return b < t.b;
    }
} l[N];

int main()
{
    for (int x1 = 0; x1 < 20; x1 ++ )
        for (int y1 = 0; y1 < 21; y1 ++ )
            for (int x2 = 0; x2 < 20; x2 ++ )
                for (int y2 = 0; y2 < 21; y2 ++ )
                    if (x1 != x2) // 先不考虑斜率不存在的直线 
                    {
                        double k = (double)(y2 - y1) / (x2 - x1);
                        double b = y1 - k * x1;
                        l[n ++ ] = {k, b}; 
                    }

    sort(l, l + n);
    int res = 1; // 这里比较的时候是从第二条直线开始比较的,每次都和前面的直线比较,所以res = 1 
    for (int i = 1; i < n; i ++ ) // 从第二条直线开始
        if (fabs(l[i].k - l[i - 1].k) > 1e-8 || fabs(l[i].b - l[i - 1].b) > 1e-8)
            res ++ ;
    cout << res + 20 << endl; // 最后加上斜率不存在的20条直线 

    return 0; 
}

作者:动S名C

答案:40257

  • D.货物摆放

题目描述
小蓝有一个超大的仓库,可以摆放很多货物。
现在,小蓝有 n 箱货物要摆放在仓库,每箱货物都是规则的正方体。小蓝
规定了长、宽、高三个互相垂直的方向,每箱货物的边都必须严格平行于长、
宽、高。
小蓝希望所有的货物最终摆成一个大的立方体。即在长、宽、高的方向上
分别堆 L、W、H 的货物,满足 n = L × W × H。
给定 n,请问有多少种堆放货物的方案满足要求。
例如,当 n = 4 时,有以下 6 种方案:1×1×4、1×2×2、1×4×1、2×1×2、
2 × 2 × 1、4 × 1 × 1。
请问,当 n = 2021041820210418 (注意有 16 位数字)时,总共有多少种
方案?

问题分析
n = a * b * c
a,b,c都是n的约数
先算n的约数,三层循环找答案

C++ 代码

#include <iostream>
#include <cstring>
#include <algorithm>
#include <vector>

using namespace std;

typedef long long LL;

int main()
{
    LL n;
    cin >> n;
    vector<LL> d; // 存n的约数
    for (LL i = 1; i * i <= n; i ++ )
        if (n % i == 0)
        {
            d.push_back(i);
            if (n / i != i) d.push_back(n / i);
        }

    int res = 0;
    for (auto a : d)
        for (auto b : d)
            for (auto c : d)
                 if (a * b * c == n)
                    res ++ ;
    cout << res << endl;
    return 0;
}

作者:动S名C

答案:2430

 

  • E.路径

题目描述
小蓝学习了最短路径之后特别高兴,他定义了一个特别的图,希望找到图
中的最短路径。
小蓝的图由 2021 个结点组成,依次编号 1 至 2021。
对于两个不同的结点 a, b,如果 a 和 b 的差的绝对值大于 21,则两个结点
之间没有边相连;如果 a 和 b 的差的绝对值小于等于 21,则两个点之间有一条
长度为 a 和 b 的最小公倍数的无向边相连。
例如:结点 1 和结点 23 之间没有边相连;结点 3 和结点 24 之间有一条无
向边,长度为 24;结点 15 和结点 25 之间有一条无向边,长度为 75。
请计算,结点 1 和结点 2021 之间的最短路径长度是多少。

问题分析
裸的最短路问题(这里用的是spfa)

C++ 代码

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 2200, M = N * 50;

int n;
int h[N], e[M], w[M], ne[M], idx;
int q[N], dist[N];
bool st[N];

int gcd(int a, int b)  // 欧几里得算法
{
    return b ? gcd(b, a % b) : a;
}

void add(int a, int b, int c)  // 添加一条边a->b,边权为c
{
    e[idx] = b, w[idx] = c, ne[idx] = h[a], h[a] = idx ++ ;
}

void spfa()  // 求1号点到n号点的最短路距离
{
    int hh = 0, tt = 0;
    memset(dist, 0x3f, sizeof dist);
    dist[1] = 0;
    q[tt ++ ] = 1;
    st[1] = true;

    while (hh != tt)
    {
        int t = q[hh ++ ];
        if (hh == N) hh = 0;
        st[t] = false;

        for (int i = h[t]; i != -1; i = ne[i])
        {
            int j = e[i];
            if (dist[j] > dist[t] + w[i])
            {
                dist[j] = dist[t] + w[i];
                if (!st[j])     // 如果队列中已存在j,则不需要将j重复插入
                {
                    q[tt ++ ] = j;
                    if (tt == N) tt = 0;
                    st[j] = true;
                }
            }
        }
    }
}

int main()
{
    n = 2021;
    memset(h, -1, sizeof h);
    for (int i = 1; i <= n; i ++ )
        for (int j = max(1, i - 21); j <= min(n, i + 21); j ++)
        {
            int d = gcd(i, j);
            add(i, j, i * j / d);
        }

    spfa();
    printf("%d\n", dist[n]);
    return 0;
}

作者:动S名C
  • F:时间显示

小蓝要和朋友合作开发一个时间显示的网站。

在服务器上,朋友已经获取了当前的时间,用一个整数表示,值为从 1970 年 11 月 11 日 00:00:00到当前时刻经过的毫秒数。

现在,小蓝要在客户端显示出这个时间。

小蓝不用显示出年月日,只需要显示出时分秒即可,毫秒也不用显示,直接舍去即可。

给定一个用整数表示的时间,请将这个时间对应的时分秒输出。

输入格式

输入一行包含一个整数,表示时间。

输出格式

输出时分秒表示的当前时间,格式形如 HH:MM:SS,其中 HH 表示时,值为 00 到 2323,MM 表示分,值为 00 到 59,SS 表示秒,值为 00 到 59。

时、分、秒不足两位时补前导 00。

数据范围

对于所有评测用例,给定的时间为不超过 10^{18} 的正整数。

输入样例1:

46800999

输出样例1:

13:00:00

输入样例2:

1618708103123

输出样例2: 

01:08:23
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
int main()
{
	ll n;
	cin>>n;
	n/=1000;
	n%=86400;
	ll a=n/3600,b=n/60%60,c=n%60;
	printf("%02d:%02d:%02d",a,b,c);
}
  • G:砝码称重

你有一架天平和N个砝码,这 N个砝码重量依次是 W1,W2,⋅⋅⋅,WN。

请你计算一共可以称出多少种不同的正整数重量?

注意砝码可以放在天平两边。

输入格式

输入的第一行包含一个整数 N。

第二行包含 N 个整数:W1,W2,W3,⋅⋅⋅,WNW1,W2,W3,···,WN。

输出格式

输出一个整数代表答案。

数据范围

对于 50%的评测用例,1≤N≤15。
对于所有评测用例,1≤N≤100,N 个砝码总重不超过 10^5

输入样例:

3
1 4 6

输出样例:

10

样例解释

能称出的 1010 种重量是:1、2、3、4、5、6、7、9、10、111、2、3、4、5、6、7、9、10、11。

1 = 1;
2 = 6 − 4 (天平一边放 6,另一边放 4);
3 = 4 − 1;
4 = 4;
5 = 6 − 1;
6 = 6;
7 = 1 + 6;
9 = 4 + 6 − 1;
10 = 4 + 6;
11 = 1 + 4 + 6。
#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;
// 重量的范围是-m到m,m是所有砝码的和
const int N = 110, M = 200010, B = M / 2; // B是偏移量

int n, m; // n个砝码,m是所有砝码的和
int w[N]; // 每个砝码的重量
bool f[N][M]; // f(i, j) 表示在前i个砝码中,总重量为j的所有方案的个数

int main()
{
    cin>>n;
    for (int i = 1; i <= n; i ++ ) scanf("%d", &w[i]), m += w[i];

    f[0][B] = true; // 初始化;
    for (int i = 1; i <= n; i ++ )
        for (int j = -m; j <= m; j ++ )
        {
            f[i][j + B] = f[i - 1][j + B];
            if (j - w[i] >= -m) f[i][j + B] |= f[i - 1][j - w[i] + B];
            if (j + w[i] <= m) f[i][j + B] |= f[i - 1][j + w[i] + B];
        }

    int res = 0;
    for (int j = 1; j <= m; j ++ ) // 看一下所有正整数重量能否取到
        if (f[n][j + B])
            res ++ ;
    cout<<res<<endl;
    return 0;
}

作者:动S名C
  • E:杨辉三角形

如果我们按从上到下、从左到右的顺序把所有数排成一列,可以得到如下数列:

1, 1, 1, 1, 2, 1, 1, 3, 3, 1, 1, 4, 6, 4, 1, ...

给定一个正整数 N,请你输出数列中第一次出现 NN 是在第几个数?

输入格式

输入一个整数 N。

输出格式

输出一个整数代表答案。

数据范围

对于 20% 的评测用例,1≤N≤10;
对于所有评测用例,1≤N≤10^{9}

输入样例:

6

输出样例:

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

typedef long long LL;

int n;

LL C(int a, int b) // 求排列组合C
{
    LL res = 1;
    for (int i = a, j = 1; j <= b; i -- , j ++ )
    {
        res = res * i / j;
        if (res > n) return res;
    }
    return res;
}

bool check(int k) // 二分找到大于n的最小的数
{
    LL l = k * 2, r = max((LL)n, l);
    while (l < r)
    {
        LL mid = l + r >> 1;
        if (C(mid, k) >= n) r = mid;
        else l = mid + 1;
    }
    if (C(r, k) != n) return false;

    cout << r * (r + 1) / 2 + k + 1 << endl;

    return true;
}

int main()
{
    cin >> n;
    for (int k = 16; ; k -- ) // 从第16个斜行(最下面的斜行)开始遍历
        if (check(k))
            break;
    return 0;
}

作者:动S名C
  • I:双向排序

给定序列 (a1,a2,⋅⋅⋅,an)=(1,2,⋅⋅⋅,n),即 ai=i。

小蓝将对这个序列进行 m 次操作,每次可能是将 a1,a2,⋅⋅⋅,aqi降序排列,或者将 aqi,aqi+1,⋅⋅⋅,an升序排列。

请求出操作完成后的序列。

输入格式

输入的第一行包含两个整数 n,m,分别表示序列的长度和操作次数。

接下来 m行描述对序列的操作,其中第 i 行包含两个整数 pi,qi 表示操作类型和参数。当 pi=0时,表示将 a1,a2,⋅⋅⋅,aqi降序排列;当 pi=1时,表示将 aqi,aqi+1,⋅⋅⋅,an 升序排列。

输出格式

输出一行,包含 n 个整数,相邻的整数之间使用一个空格分隔,表示操作完成后的序列。

数据范围

对于 30%的评测用例,n,m≤1000;
对于 60%的评测用例,n,m≤5000;
对于所有评测用例,1≤n,m≤10^{5},0≤pi≤1,1≤qi≤n。

输入样例:

3 3
0 3
1 2
0 2

输出样例:

3 1 2
#include <iostream>
#include <cstring>
#include <algorithm>

#define x first
#define y second

using namespace std;

typedef pair<int, int> PII;

const int N = 100010;

int n, m;
PII stk[N];
int ans[N];

int main()
{
    cin>>n>>m;
    int top = 0;
    while (m -- )
    {
        int p, q;
        cin>>p>>q;
        if (!p)
        {
            while (top && stk[top].x == 0) q = max(q, stk[top -- ].y);
            while (top >= 2 && stk[top - 1].y <= q) top -= 2;
            stk[ ++ top] = {0, q};
        }
        else if (top)
        {
            while (top && stk[top].x == 1) q = min(q, stk[top -- ].y);
            while (top >= 2 && stk[top - 1].y >= q) top -= 2;
            stk[ ++ top] = {1, q};
        }
    }
    int k = n, l = 1, r = n;
    for (int i = 1; i <= top; i ++ )
    {
        if (stk[i].x == 0)
            while (r > stk[i].y && l <= r) ans[r -- ] = k -- ;
        else 
            while (l < stk[i].y && l <= r) ans[l ++ ] = k -- ;
        if (l > r) break;
    }
    if (top % 2)
        while (l <= r) ans[l ++ ] = k -- ;
    else 
        while (l <= r) ans[r -- ] = k -- ;
    for (int i = 1; i <= n; i ++ )
        cout<<ans[i]<<" ";
    return 0;
}

作者:动S名C
//sort可骗60%
/*#include<bits/stdc++.h>
using namespace std;
bool cmp(int a,int b)
{
	return a>b;
}
int main()
{
	int n,m,p[100005],q[100005],a;
	cin>>n>>m;
	for(int i=1;i<=n;i++)
	{
		q[i]=i;
	}
	for(int i=0;i<m;i++)
	{
		cin>>p[i]>>a;
		if(p[i]==0)
		{
			sort(q+1,q+1+a,cmp);
		}
		if(p[i]==1)
		{
			sort(q+a,q+1+n);
		}
	}
	for(int i=1;i<=n;i++)
	{
		cout<<q[i]<<" ";
	}
return 0;
 } */
  •  J:括号序列

给定一个括号序列,要求尽可能少地添加若干括号使得括号序列变得合法,当添加完成后,会产生不同的添加结果,请问有多少种本质不同的添加结果。

两个结果是本质不同的是指存在某个位置一个结果是左括号,而另一个是右括号。

例如,对于括号序列 (((),只需要添加两个括号就能让其合法,有以下几种不同的添加结果:()()()、()(())、(())()、(()()) 和 ((()))

输入格式

输入一行包含一个字符串 ss,表示给定的括号序列,序列中只有左括号和右括号。

输出格式

输出一个整数表示答案,答案可能很大,请输出答案除以 10^{9}+7的余数。

数据范围

对于 40%的评测用例,|s|≤200。
对于所有评测用例,1≤|s|≤5000。

输入样例:

((()

输出样例:

5
#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

typedef long long LL;

const int N = 5010, MOD = 1e9 + 7;

int n;
char str[N];
LL f[N][N];

LL work()
{
    memset(f, 0, sizeof(f));
    f[0][0] = 1;
    for (int i = 1; i <= n; i ++ )
        if (str[i ] == '(')
        {
            for (int j = 1; j <= n; j ++ )
                f[i][j] = f[i - 1][j - 1];
        }
        else
        {
            f[i][0] = (f[i - 1][0] + f[i - 1][1]) % MOD;
            for (int j = 1; j <= n; j ++ )
                f[i][j] = (f[i - 1][j + 1] + f[i][j - 1]) % MOD;
        }

    for (int i = 0; i <= n; i ++ )
        if (f[n][i])
            return f[n][i];
    return -1;
}

int main()
{
    scanf("%s", str + 1);
    n = strlen(str + 1);
    LL l = work();
    reverse(str + 1, str + n + 1);
    for (int i = 1; i <= n; i ++ )
        if (str[i] == '(') str[i] = ')';
        else str[i] = '(';
    LL r = work();
    printf("%lld\n", l * r % MOD);
    return 0;
}

作者:动S名C
  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值