AtCoder Beginner Contest 367 Solution
A - Shout Everyday
高桥从B睡到C,如果在A时刻时,他醒着就会尖叫,问他是不是会尖叫。
说白就是问你A在不在BC两个时刻之间。
#include<bits/stdc++.h>
using namespace std;
int A, B, C;
int main() {
cin >> A >> B >> C;
if ((A - B + 24) % 24 > (C - B + 24) % 24) {
cout << "Yes\n";
} else {
cout << "No\n";
}
return 0;
}
也可以从B循环到C。
B - Cut .0
给定一个小数,去除后导0,如果删完小数部分则还要把小数点删了。
#include <bits/stdc++.h>
using namespace std;
int main() {
string s;
cin >> s;
while (s.back() == '0')
s.pop_back();
if (s.back() == '.')
s.pop_back();
cout << s << '\n';
return 0;
}
C - Enumerate Sequences
要求输出所有符合要求的数组a,要求数组的和是k的倍数, 并且每个数的范围是1~ri。
枚举
#include <bits/stdc++.h>
using namespace std;
const int N = 201000;
int n, k, r[N], b[N];
void dfs(int x, int s) {
if (x == n + 1) {
if (s == 0) {
for (int i = 1; i <= n; i++) printf("%d ", b[i]);
puts("");
}
} else {
for (int i = 1; i <= r[x]; i++) {
b[x] = i;
dfs(x + 1, (s + i) % k);
}
}
}
int main() {
scanf("%d%d", &n, &k);
for (int i = 1; i <= n; i++) scanf("%d", &r[i]);
dfs(1, 0);
return 0;
}
D - Pedometer
环形湖上有n个点,给定ai表示从第i个点顺时针走到第i+1个点的时间。
问有多少对(s,t),满足顺时针从s走到t的时间是m的倍数。
思路
先考虑s<=t的情况,那么就是问有几段a数组的和是m的倍数,我们可以对前缀和取模,那么前缀和的模相等的点之间的和肯定是模m为0的,也就是问统计前缀和模相等的对数,具体做法:开一个cnt数组用于记数前面前缀和的模,然后枚举终点,把cnt里有多少个起点加进ans里即可。
那环形怎么办,可以小技巧:把环解成链,比如1234的环做成12341234,然后做一个类似于滑动窗口的操作,我们在走到第五位的时候把第一位在cnt里删掉,那么我们统计的就是第二位到第五位的合法s和t。
#include <bits/stdc++.h>
#define LL long long
using namespace std;
int main() {
int n, m;
cin >> n >> m;
vector<int> a(n);
for (auto& x : a)
cin >> x;
vector<int> cnt(m, 0);
LL ans = 0;
LL presum = 0;
for (int i = 0; i < n; ++i) {
ans += cnt[presum % m];
cnt[presum % m]++;
presum += a[i];
}
LL sum = presum;
presum = 0;
for (int i = 0; i < n; ++i) {
cnt[presum % m]--;
ans += cnt[sum % m];
sum += a[i];
presum += a[i];
}
cout << ans << '\n';
return 0;
}
E - Permute K times
给定数组x和a,一共进行k次操作,每次操作做一个新数组bi=axi,然后把a数组替换为b数组,问k此操作之后的数组。
思路
需要知道的常识:如果x是一个1~n的排列的话,对于像这样的变化,如果把x的每次操作画成图的话,x一定是由很多环构成的,每个环的大小必定不大于n。
不是排列的话,那么他其实是一个基环树森林,但我们不用管他,因为那些经过一次操作到达相同点的点以后所有的操作都是一样的,我们不用处理。
这题其实是一道非常模板的倍增,k非常大,我们处理一个二维数组run[64][n]
,其中run[i][j]
表示第j个位置走$2^i$步之后会到哪里,这个数组非常好处理,因为第j个点走$2^i$步等于他从先走$2^{i-1}$步到达的地方再走$2^{i-1}$步,写成转移就是run[i][j] = run[i - 1][run[i - 1][j]]
,这就是倍增的基本思想。
同样,根据我们这个先走几步,再走几步的思想,我们直接把k分解成二进制位,就能直接调用run数组每次走2的整次幂倍的步数,找到每个点k步后的目标了。
#include <bits/stdc++.h>
#define LL long long
using namespace std;
int run[64][200005], a[200005];
int main() {
int n;
LL k;
cin >> n >> k;
for (int i = 0; i < n; i++) {
int v;
cin >> v;
--v;
run[0][i] = v;
}
for (int i = 0; i < n; i++)
cin >> a[i];
for (int i = 1; i < 64; i++) {
for (int j = 0; j < n; j++) {
run[i][j] = run[i - 1][run[i - 1][j]];
}
}
for (int i = 0; i < n; i++) {
LL cnt = k;
int cur = i;
for (int j = 0; j < 64; j++) {
if (cnt & (1LL << j)) {
cur = run[j][cur];
}
}
cout << a[cur] << " ";
}
return 0;
}
F - Rearrange Query
给定两个长度为n的数组a和b,并且有Q次查询,第i次查询给定li,ri,Li,Ri,问a和b里的对应区间能不能通过重新排列的方式使两者匹配。
如果要两段一样,那么两段的和必须一样,如果只用前缀和判断两段和是否一样会被构造的数据卡掉,所以我们对每一个数都给他随机映射到另一个数上,这样求和的方式就不容易被卡掉了。
这就是哈希,冲突率在这题里是不高的。
#include <bits/stdc++.h>
#define ll long long
using namespace std;
mt19937_64 mrand(random_device{}());
const int N=201000;
int n,q;
ll hs[N],sa[N],sb[N];
int main() {
scanf("%d%d",&n,&q);
for(int i=1;i<=n;i++) hs[i]=mrand();
for(int i=1;i<=n;i++) {
int a;
scanf("%d",&a);
sa[i]=sa[i-1]+hs[a];
}
for(int i=1;i<=n;i++) {
int a;
scanf("%d",&a);
sb[i]=sb[i-1]+hs[a];
}
while(q--){
int l1,r1,l2,r2;
scanf("%d%d%d%d",&l1,&r1,&l2,&r2);
puts((sa[r1]-sa[l1-1]==sb[r2]-sb[l2-1])?"Yes":"No");
}
return 0;
}