2022ACM寒假第一周训练部分题目代码
在比赛结束后可以查看其他人的代码,
这样的是可以查看, 若非绿色则对方没公开。
1 周一
1.1 A 链表应用
#include <iostream>
#include <list>
using namespace std;
/*
* 朴素想法是用数组存入小孩, 然后每次出列就遍历一下数组, 找满足条件的对应下标,
* 同时用另一个数组记录每个孩子是否以及出列的状态。最坏时间复杂度为 O(n^2)
* 该题这样写是能过的, 数据范围太小。
* 不过若数据范围 N >= 10000, 必须优化查找下一个出队元素的操作
* 既然我们要对数组进行删除操作(标记为出列时, 对于后续操作来说也算删除), 也就是需要变动数组,
* 很显然有一个数据结构是线性, 且删除,增加的操作时间复杂度为O(1), 即链表
* 对于这样一个链表:
* 1 -> 2 -> 3 -> 4
* 想删除第二个元素时, 只需要让 1 跳过2, 指向3, 这样从一开始遍历时, 2就和被删除的效果是一样的:
* 1 -> 3 -> 4
* 而若数组想实现这个操作, 需要在经历一次遍历, 把2之后的数往前挪动一位
* 该题围成一圈, 可以让 最后一个元素也指向头元素, 这样从任意一点遍历都可以走完一圈。
*/
// 数组模拟链表
const int N = 100;
char e[N][16];
int ne[N], idx, head;
void init()
{
head = 0;
idx = 1;
ne[head] = 1;
}
void ArraySlove()
{
init();
int n;
cin >> n;
for (int i = 1; i <= n; i++)
{
scanf("%s", &e[idx]);
ne[idx] = idx + 1;
idx++;
}
ne[idx - 1] = 1; // 将尾结点连到头结点
// 遍历输出, 对-1按位取反的结果就是0, 故可以用 ~i 代替 i != -1
// for (int i = head; ~i; i = ne[i])
// cout << e[i] << endl;
int w, s;
scanf("%d,%d", &w, &s);
for (int i = w - 1, count = 0, res = 0;;)
{
if (res == n - 1) // 形成自环, 即只剩下一个元素
{
printf("%s\n", e[ne[i]]);
break;
}
if (count == s - 1)
{
int next = ne[i];
printf("%s\n", e[next]);
ne[i] = ne[next]; // 将下一个元素删去
count = 0;
res++;
continue;
}
else
{
i = ne[i];
count++;
}
}
return;
}
int main()
{
ArraySlove();
return 0;
}
1.2 B 状态模拟题
Edge浏览器支持右键翻译整个网页, 虽然不太准确, 不过相信你根据样例也能搞清楚这题什么意思。
给定一个整数和一个字符串, 整数代表有几个位置, 字符第一次出现表示占据一个位置, 第二次出现表示离开, 让出一个位置。
求最后有多少个字符第一次出现时没有位置。
用st数组表示当前字符的出现情况, 0为未出现, 1为出现过且当时有位置, -1为出现过且当时无位置。
#include <iostream>
#include <cstring>
using namespace std;
const int N = 53; // 26个英文字母最多出现两次, 故最多为52个
char s[N];
int st[26]; // 表示当前字母是否出现。
int main()
{
int n;
while (cin >> n && n)
{
memset(st, 0, sizeof st);
cin >> s;
int cnt = 0, m = strlen(s); // cnt无法找到床位的人数
for (int i = 0; i < m; i++)
{
int t = s[i] - 'A'; // 将字符映射到对应的整数
// 为什么可以这样做是因为他们的ASCII码连续, 这种操作在以后的字符串处理很常用, 建议熟练掌握
if (!st[t] && !n) // 如果当前字符第一次出现, 且没床位
{
cnt++;
st[t] = -1; // 设成-1, 这样后面再来一个也不会处理
}
else if (!st[t] && n)
{
n--;
st[t] = 1;
}
else if (st[t] == 1)
n++;
}
if (!cnt)
cout << "All customers tanned successfully.\n";
else
cout << cnt << " customer(s) walked away.\n";
}
return 0;
}
1.3 C 队列应用
这题放到第一天确实有点超纲, 不过学完一周的你回来再看应该会很轻松。
题目要求模拟一个任务处理系统, 输入一系列任务, 从头开始依次处理, 若当前任务的优先级不是最高, 则将其放到末尾;若为最高则直接处理。
让求的是其中一个任务被处理完时经过了多少时间。
既然有从开头剔除元素, 有从末尾加入元素, 那么我们可以把这任务处理队列看成一个队列(废话
主要问题在于怎么判断当前任务是不是最高优先级。
很多同学都用的是最大堆来做啊(priority_queue 优先队列, 也就是堆), 让它自行维护一个最大值。其实没必要, 我们并不需要向其中乱序加入一个元素, 所用到的功能也就剔除最大值, 然后求当前最大值.
可以先将优先级存在数组a
中, 用idx
代表当前的最高优先级, 把a
数组从大到小排序, idx=0
那么a[idx]
就代表最大的优先级, 剔除一个元素后, 我们要找的是次大优先级, 而a
已经是从大到小排序, 那么下一个元素a[idx+1]
就是要找的次大元素。
#include <iostream>
#include <queue>
#include <algorithm>
using namespace std;
const int N = 5060, M = 110; // 若优先级呈严格递增序列, 那么第一遍会把前n-1项接到后面, 第二次会把 n-2项接到后面
// 故我们队列最多会使用 n + n-1 + n-2 + n-3 ... + 1 = n(n+1)/2
// 带入题目范围得 N = 5050, 这是最坏条件下使用的队列大小
// 而若使用STL库 #include <queue> 则不需要考虑大小问题。 不过有可能因为卡常数而TLE(不知道啥事卡常的自己百度:monocle_face:)
// 这样是有点浪费, 更好的解决方法是循环队列
int q[N], hh, tt; // 双端队列, 可以从队头hh加入/删除元素, 也可以从队尾tt加入/删除元素
int a[M]; // 将任务进行优先级排序
int n, m;
int main()
{
int T;
cin >> T;
while (T--)
{
cin >> n >> m;
hh = 0, tt = -1; // 初始化队列, -1是因为加入元素用的是++tt
for (int i = 0; i < n; i++)
{
cin >> q[++tt];
a[i] = q[tt];
}
sort(a, a + n, greater<int>()); // sort函数第三个参数为指定排序方式,
// 默认为less<int>(), 即从小到大排序, 这里的greater<int>()则为从大到小排序, 你也可以手写一个check()函数
int idx = 0, res = 0; // 当前的最高优先级为 a[idx], 等待时间为res
while (hh <= tt) // 若队列不为空
{
int t = q[hh++]; // 取当前队头元素并出队
if (t >= a[idx]) // 若大于当前最大优先级, 则可以出队并执行
{
if (hh - 1 == m) // 若当前下标为老师的计算任务, 则输出结果并退出循环
{
cout << res + 1 << endl;
break;
}
// 否则就res++, 继续循环
res++;
idx++; // 当前任务处理后, 处理次高优先级
}
else
{
if (hh - 1 == m) // 若当前下标为老师的计算任务, 且优先级低, 需要回到队尾
{
q[++tt] = t;
m = tt; // 将标识着老师任务的下标一并更新
} // 若不是则直接加即可
else
{
q[++tt] = t;
}
}
}
}
return 0;
}
2 周二
2.1 C 队列安排
假如该题只会插入到右边, 那么只需要一个ne就能解决, 但这里还会插入到左边, 因此需要知道该点之前的信息, 这是单链表无法提供的, 故这里需要使用双链表。
记得用st数组来标记已经被删除了的元素, 避免重复删除
#include <iostream>
using namespace std;
const int N = 1e5 + 10;
int e[N], l[N], r[N], idx;
bool st[N];
void init()
{
l[1] = 0; // 初始右节点
r[0] = 1; // 初始左节点
idx = 2;
}
void add(int i, int x) // 插入到编号为i的元素后面
{
e[idx] = x;
l[idx] = i; // 新加入的点的左端点编号为 i
r[idx] = r[i]; // 新加入的点的右端点编号为 r[i], 即i节点的右端点
l[r[i]] = idx; // i节点的右端点, 也就是新加入点的右端点, 将其左端点从i更新为x
r[i] = idx++;
}
void del(int x)
{
l[r[x]] = l[x];
r[l[x]] = r[x];
}
int main()
{
init();
int n;
cin >> n;
add(l[0], 1); // 把
for (int i = 2; i <= n; i++)
{
int k, p;
cin >> k >> p;
if (p)
add(k + 1, i);
else
add(l[k + 1], i);
}
cin >> n;
while (n--)
{
int x;
cin >> x;
if (!st[x])
del(x + 1);
st[x] = true;
}
// for (int i = 0; i < idx; i++)
// cout << l[i] << " " << r[i] << " " << e[i] << endl;
for (int i = r[0]; i != 1; i = r[i])
cout << e[i] << " ";
return 0;
}
3 周三
3.1 B 扩号匹配问题
每个左括号与距离最右边的括号匹配, 像这种匹配问题和计算表达式问题, 都是用栈来写。
当遇到左括号时, 将其下标入栈, 遇到右括号时, 将当前栈顶元素出栈。
那么遍历完后, 若栈中还剩下左括号, 就是无法匹配的左括号;若遇到右括号时栈空, 就是无法匹配的右括号。
#include <iostream>
#include <cstring>
#include <algorithm>
#include <string>
#include <cmath>
using namespace std;
const int N = 110;
char a[N];
char res[N];
int stack[N], top;
int main()
{
while (cin >> a)
{
top = -1;
int n = strlen(a);
for (int i = 0; i < n; i++)
{
res[i] = ' ';
if (top == -1 && a[i] == ')')
res[i] = '?';
else if (a[i] == '(')
stack[++top] = i;
else if (a[i] == ')')
top--;
}
for (int i = 0; i <= top; i++)
res[stack[i]] = '$';
cout << a << endl
<< res << endl;
}
return 0;
}
3.2 C 士兵队列训练问题
和周一A题有点像, 不过比那题更简单些, 不是环形了。
#include <iostream>
#include <cstring>
using namespace std;
const int N = 6e3;
int ne[N], e[N], idx, head;
int n, size;
bool flag = true;
int cnt;
void init()
{
head = -1;
idx = 1;
memset(ne, 0, sizeof ne);
memset(e, 0, sizeof e);
flag = true;
}
void add(int x)
{
e[idx] = x;
ne[idx] = head;
head = idx++;
}
void del(int k) // 删除k+1项
{
ne[k] = ne[ne[k]];
}
bool check(int x)
{
if (flag)
{
if (x % 2 == 0)
return true;
}
else
{
if (x % 3 == 0)
return true;
}
return false;
}
int main()
{
int T;
cin >> T;
while (T--)
{
init();
cin >> n;
for (int i = n; i >= 1; i--) // 逆序存数, 这样枚举时便是正序
add(i);
int prev = head;
while (n > 3)
{
cnt = 1;
for (int i = head; ~i; prev = i, i = ne[i], cnt++)
{
if (check(cnt))
{
del(prev);
cnt = 0;
n--;
}
}
flag = !flag;
}
for (int i = head; ~i; i = ne[i])
cout << e[i] << " \n"[ne[i] == -1];
}
return 0;
}
4 周四
4.1 B 最小字典序字符串
题目概括一下是给一个串s, 提供两个空串t,u。通过一下两个操作:
- s的第一个字符移动到t末尾
- t的最后一个字符移动到u末尾
求出能得到的最小字典序。
贪心+栈的题目, 对于结果u串, 当前位置上的字符要么从s串中来, 要么从t串中来。
若u任意位置上的选择都是最小的, 那么总的字典序也是最小的
证明:
u1: ... ... ... a ... ...
u2: ... ... ... b ... ...
若 a 的字典序小于 b 的字典序, 那么根据字典序的定义, u1的字典序就小于u2的字典序。u1才是正确结果。
当前位置可以从s串和t串中来:
- 若s串存在能选的最小元素小于t串末尾, 则将其以及之前的字符都加入到t串, 并把该元素从t串末尾移到u串末尾
- 若大于等于, 则把t串末尾放到u串末尾
#include <iostream>
#include <cstring>
#include <algorithm>
#include <string>
#include <cmath>
using namespace std;
const int N = 1e5 + 10;
char s[N], u[N], t[N];
int top_u = -1, topt = -1;
char min_c[N];
int n;
int main()
{
cin >> s;
n = strlen(s);
min_c[n] = 'z';
for (int i = n - 1; i >= 0; i--)
min_c[i] = min(s[i], min_c[i + 1]);
for (int i = 0, j = 0; i < n; i = j)
{
if (topt == -1 || t[topt] > min_c[i])
{
for (j; s[j] != min_c[i] && j < n; j++)
t[++topt] = s[j];
u[++top_u] = min_c[i];
j++;
}
else
u[++top_u] = t[topt--];
}
while (topt != -1)
u[++top_u] = t[topt--];
cout << u << endl;
return 0;
}
5 周五
5.1 B Golden Sword
是一道单调队列优化的动态规划问题, 严重超纲了, 已让出题人改悔。
这题不必做。