- 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。
数据范围
对于所有评测用例,给定的时间为不超过 的正整数。
输入样例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 个砝码总重不超过 。
输入样例:
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≤。
输入样例:
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≤,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,表示给定的括号序列,序列中只有左括号和右括号。
输出格式
输出一个整数表示答案,答案可能很大,请输出答案除以 +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