2089A - Simple Permutation

从小往大顺着选,如果不是素数就选个稍微大一点的让结果是素数。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
#include <bits/stdc++.h>
using namespace std;
using ll = long long;

constexpr int N = 5e5 + 5;

vector<int> prime;
int mint[N]; // 最小素因子

bool is_prime(int x) {
return mint[x] == x;
}

signed main() {
ios::sync_with_stdio(false);
cin.tie(nullptr), cout.tie(nullptr);

for (int i = 2; i < N; i++) {
if (!mint[i]) mint[i] = i, prime.push_back(i);
for (auto& p : prime) {
if (1ll * i * p >= N) break;
mint[i * p] = p;
if (p == mint[i]) break;
}
}

int J;
cin >> J;

while (J--) {
int n;
cin >> n;

set<int> s;
for (int i = 1; i <= n; i++) {
s.insert(i);
}

vector<int> res;
ll sum = 0;
for (int i = 1; i <= n; i++) {
int x = *s.begin();
int now = (sum + x - 1) / i + 1;
while (!is_prime(now)) {
now++;
}
int need = i * (now - 1) + 1 - sum;
auto it = s.lower_bound(need);
if (it == s.end()) {
it--;
}
int x = *it;
sum += x;
s.erase(x);
res.push_back(x);
}

for (int i = 0; i < n; i++) {
cout << res[i] << ' ';
}
cout << endl;

// sum = 0;
// for (int i = 0; i < n; i++) {
// sum += res[i];
// cout << (sum - 1) / (i + 1) + 1 << ' ';
// }
// cout << endl;
}

return 0;
}

2089B1 - Canteen (Easy Version)

每操作一次(ai,bi)(a_{i},b_{i}),序列a,ba,b 中 0 的个数就会增多一个。因此真正有用的操作数(不是题目中的轮数)不会超过2n2n

提供一种常数巨大的写法,用 bitset 存a,ba,b 中仍然不是 0 的位置,将这两个 bitset 作与,就能得到本轮所有需要操作的(ai,bi)(a_{i},b_{i}) 了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
#include <bits/stdc++.h>

using namespace std;
using ll = long long;

constexpr int N = 2e5 + 5;

bitset<N> A, B;

signed main() {
ios::sync_with_stdio(false);
cin.tie(nullptr), cout.tie(nullptr);

int J;
cin >> J;

while (J--) {
int n, k;
cin >> n >> k;

vector<int> a(n);
for (int i = 0; i < n; i++) {
cin >> a[i];
}

vector<int> b(n);
for (int i = 0; i < n; i++) {
cin >> b[i];
}

for (int i = 0; i < n; i++) {
A[i] = 1;
B[i] = 1;
}

int cnt = 0;

for (int d = 0; d < n; d++) {
auto C = A & B;
for (int i = C._Find_first(); i != N; i = C._Find_next(i)) {
int& x = a[(i - d + n) % n];
int& y = b[i];
if (x > y) {
x -= y;
y = 0;
B[i] = 0;
} else {
y -= x;
x = 0;
A[i] = 0;
cnt++;
}
}
if (cnt == n) {
cout << d + 1 << endl;
break;
}
A <<= 1;
A[0] = A[n];
A[n] = 0;
}
}

return 0;
}

2089C1 - Key of Like (Easy Version)

最优方法是对每个锁孔,拿所有钥匙挨个试一遍。

先设nn\to \infty,设f(,i)f(\ell,i) 表示\ell 把锁时第ii 个人的答案。

考虑递推,已经算出1\ell-1 把锁时的答案f(1,1),f(1,2),f(\ell-1,1),f(\ell-1,2),\dots。如果增加一把锁,不妨设第一个人先尝试这把锁。

1\cfrac{1}{\ell} 的概率第一个人一次成功,那么他后面的人就是1\ell-1 的情况,

{f(,1)=1,f(,2)=1f(1,1),f(,3)=1f(1,2),\begin{cases} f(\ell,1)=\cfrac{1}{\ell}, \\ f(\ell,2)=\cfrac{1}{\ell}f(\ell-1,1), \\ f(\ell,3)=\cfrac{1}{\ell}f(\ell-1,2), \\ \cdots \end{cases}

1\cfrac{1}{\ell} 的概率第一个人失败,第二个人成功,那么他后面的人也是1\ell-1 的情况,

{f(,1)=0,f(,2)=1,f(,3)=1f(1,1),f(,4)=1f(1,2),\begin{cases} f(\ell,1)=0, \\ f(\ell,2)=\cfrac{1}{\ell}, \\ f(\ell,3)=\cfrac{1}{\ell}f(\ell-1,1), \\ f(\ell,4)=\cfrac{1}{\ell}f(\ell-1,2), \\ \cdots \end{cases}

这样总共有\ell 种情况,全部累加。

不难写出这样的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
vector<Z> res(l * l);
for (int x = 1; x <= l; x++) {
Z inv = Z(1) / x;
vector<Z> nres(l * l);
for (int i = 0; i < x; i++) {
nres[i] += inv;
for (int j = 0; j + i + 1 < x * x; j++) {
nres[j + i + 1] += res[j] * inv;
}
}
res = nres;
}
vector<Z> ans(n);
for (int i = 0; i < l * l; i++) {
ans[i % n] += res[i];
}
for (int i = 0; i < n; i++) {
cout << ans[i] << " ";
}
cout << endl;

复杂度为O(3)\operatorname{O}(\ell^{3}),需要优化。

所需答案的下标是模nn 的,因此上述所有下标均可以在模nn 下运算:

1
2
3
4
5
6
7
8
9
10
11
12
vector<Z> res(n);
for (int x = 1; x <= l; x++) {
Z inv = Z(1) / x;
vector<Z> nres(n);
for (int i = 0; i < x; i++) {
nres[i % n] += inv;
for (int j = 0; j < n; j++) {
nres[(j + i + 1) % n] += res[j] * inv;
}
}
res = nres;
}

复杂度为O(n2)\operatorname{O}(n^{}\ell^{2}),仍需要优化。

发现第二层循环的xx,有大部分的内层循环完全相同,把这层循环也在模nn 下运算:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
vector<Z> res(n);
for (int x = 1; x <= l; x++) {
Z inv = Z(1) / x;
vector<Z> nres(n);
int q = x / n, r = x % n;
for (int i = 0; i < n; i++) {
nres[i % n] += inv * q;
for (int j = 0; j < n; j++) {
nres[(j + i + 1) % n] += res[j] * inv * q;
}
}
for (int i = 0; i < r; i++) {
nres[i % n] += inv;
for (int j = 0; j < n; j++) {
nres[(j + i + 1) % n] += res[j] * inv;
}
}
res = nres;
}

最终复杂度O(n2)\operatorname{O}(n^{2}\ell)