和破壁人及 wky VP 了去年的济南区域赛。最后排第 8,虽然离前 3 还有不少距离,但至少发挥没有之前两次 ICPC 预选赛那么拉跨。这次签到题很少,并且 E 题连写带调弄了一个多小时,差点都以为要寄了。感觉有几道题目很有意思,故记录一下思路。
(前面有两队是打星参加的)
E
下文约定记号 $G_i=\\{j\mid(i,j)\text{ is a magic link}\\}$
观察到 M 的大小比较蹊跷,于是猜想有 $O(NM)$ 做法。首先不考虑题目所求,想一下如何在不删点的情况下计算总分数。
考虑如下的一个交叉产生的代价如何计算:
凑了一阵均摊复杂度之后,可以得到以下的搞法:
- 枚举 a
- 按从小到大的顺序枚举与 a 有边的点 c
- 枚举所有位于 $(a,c)$ 之间的点 b
- 快速计算 $(a,c)$ 与 $(b,>c)$ 交叉产生的代价
考虑如何快速计算 $(a,c)$ 与 $(b,>c)$ 交叉产生的代价。注意到 c 是不断增加的,那么对于每个 b,形如 $(b,>c)$ 的边必然是不断减少的。
于是就想到对于每个 b,将 $G_b$ 递增排序,然后维护一个指针 $ind_b$,表示 $G_b$ 中第一个 $>c$ 的值的下标。然后就可以实时维护当前符合要求的 d 之和,就可以算出 $(a,c)$ 与 $(b,d)$ 交叉产生的代价之和。
分析上述算法的复杂度:
- 前三重循环最多会跑 $O(NM)$ 次
- 对于同一个 a,每条边最多会删一次,所以维护 $ind_b$ 的复杂度也是 $O(NM)$。
$$ans'=ans-ans_\text{包含 x}-ans_\text{包含 y}+ans_\text{包含 x 和 y}$$
中间两项是很好统计的,只需要在上述算法中把答案算到 a 头上就可以了。
最后一项不好直接算,我们分两种情况讨论:
- x 和 y 在交叉边对应的 4 个点中,是相邻的
这个也好搞,只需要在上述算法中,把答案算到 a,b 头上即可。
- x 和 y 在交叉边对应的 4 个点中,是不相邻的
这就意味着 x,y 在同一条边上,只需要在上述算法中,把答案算到 a,c 头上即可。
代码:
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
|
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
const ll MXN = 2e3 + 5;
ll n, m;
vector<ll> g[MXN];
ll sum[MXN], ind[MXN], sz[MXN];
ll redu(ll x) {
if (x > n) x -= n;
if (x > n) x -= n;
if (x > n) x -= n;
if (x > n) x -= n;
return x;
}
ll only[MXN], both[MXN][MXN];
ll edge[MXN][MXN];
int main() {
/* freopen("test.in", "r", stdin); */
ios::sync_with_stdio(0);
cin.tie(0);
cin >> n >> m;
for (ll i = 1; i <= m; i++) {
ll u, v;
cin >> u >> v;
if (v < u) swap(u, v);
g[u].push_back(v);
g[v].push_back(u + n);
g[u + n].push_back(v + n);
g[v + n].push_back(u + n + n);
}
for (ll i = 1; i <= n * 2; i++) sort(g[i].begin(), g[i].end());
ll ans = 0;
for (ll i = 1; i <= n; i++) {
for (ll j = i; j < i + n; j++) {
ind[j] = sz[j] = sum[j] = 0;
while (sz[j] < g[j].size() && g[j][sz[j]] < i + n) sum[j] += redu(g[j][sz[j]++]);
}
for (ll j : g[i]) {
for (ll k = i + 1; k < j; k++) {
while (ind[k] < sz[k] && g[k][ind[k]] <= j) sum[k] -= redu(g[k][ind[k]++]);
ll delt = (i + redu(j)) * ((sz[k] - ind[k]) * redu(k) + sum[k]);
ans += delt;
only[i] += delt;
both[i][k] += delt;
edge[i][j] += delt;
}
}
}
ll res = -1e18;
for (ll i = 1; i <= n; i++)
for (ll j = i + 1; j <= n; j++) {
res = max(res, ans / 4 - only[i] - only[j] + both[i][j] + both[j][i + n] + edge[i][j]);
/* cerr<<i<<" "<<j<<" "<<res<<endl; */
}
cout << res << endl;
return 0;
}
|
C
首先,下标的顺序是可以忽略的,最后只需要把答案乘以 $\prod cnt_i!$ 即可。
首先任意两个合法的过程中,一个人取的数集合必定都是一样的(下标可以不同)。
手玩了几个小时之后,发现一个合法取数数列的规律:
- 如果最大数出现偶数次,那么在数列中出现必定是成对出现。比如说 A 某一次偷袭,取了一个最大数,则 B 也必须要取一个最大数来应对 A 的偷袭。否则 A 的取数集合就变了。
- 如果最大数出现奇数次,那么必然在一开始的时候就被取走了一个。随后问题就被转化为出现偶数次的情况。
于是大胆猜想:
- 将取数序列中最大数去掉之后,次大数也必然满足这一规律。
这个规律感觉就很对,实际也很对,但是懒得证了。然后就能做了,假如目前知道只考虑数值为 $1\sim i$ 的数,能形成的合法取数序列个数。然后只用把 $\lfloor\frac{cnt_{i+1}}{2}\rfloor$ 对 $i+1$ 插进当前的序列即可。
代码:
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
|
#include<bits/stdc++.h>
using namespace std;
using ll=long long;
const ll MXN=2e6+5;
const ll P=998244353;
ll n;
ll cnt[MXN];
ll fac[MXN],ifac[MXN];
ll qpow(ll x,ll y){
ll r=1;
while(y){
if(y&1)r=r*x%P;
x=x*x%P,y>>=1;
}
return r;
}
ll c(ll x,ll y){
if(y>x || y<0)return 0;
return fac[x]*ifac[y]%P*ifac[x-y]%P;
}
int main(){
ios::sync_with_stdio(0);
cin.tie(0);
cin>>n;
fac[0]=ifac[0]=1;
for(ll i=1;i<MXN;i++){
fac[i]=fac[i-1]*i%P;
ifac[i]=qpow(fac[i],P-2);
}
for(ll i=1;i<=n;i++){
ll x;
cin>>x;
cnt[x]++;
}
ll tot=0,ans=1;
for(ll i=1;i<=n;i++){
ans=ans*fac[cnt[i]]%P*c(tot+cnt[i]/2,cnt[i]/2)%P;
tot+=cnt[i];
}
cout<<ans<<endl;
return 0;
}
|
D
假如斜率定了,则截距取到 $i\times k-a_i$ 的中位数时,答案最小。
盲猜一波答案是关于斜率的单谷函数,然后二分谷底就可以过了。注意二分的时候要用 double
而非 int
。否则会碰到函数里平坦的地方就寄了。
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
|
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
using ld = long double;
const ll MXN = 2e5 + 5;
const ld eps = 0.1;
ll n, arr[MXN];
ld brr[MXN];
ld cal(ld k) {
for (ll i = 1; i <= n; i++) brr[i] = arr[i] - i * k;
nth_element(brr + 1, brr + (n + 1) / 2, brr + 1 + n);
ld mid = brr[(n + 1) / 2], ans = 0;
for (ll i = 1; i <= n; i++) ans += abs(mid - brr[i]);
return ans;
}
int main() {
ios::sync_with_stdio(0);
cin.tie();
cin >> n;
for (ll i = 1; i <= n; i++) cin >> arr[i];
ld mx = 1e18 / n;
ld l = -mx, r = mx;
for (ll i = 1; i <= 100; i++) {
ld mid = (l + r) / 2;
/* cerr << mid << " " << cal(mid) << " " << cal(mid + eps) << endl; */
if (cal(mid) <= cal(mid + eps))
r = mid;
else
l = mid;
}
ll mn = 1e18;
for (ll i = l - 2; i <= l + 2; i++) {
mn = min(mn, (ll)round(cal(i)));
}
cout << mn;
return 0;
}
|
A(口胡)
由于其他题时间太多了,最后没时间写。队友告诉我只需要把每个点对应坐标存到 dfn 的位置,然后维护区间加向量,区间叉乘,单点求值即可。
M(口胡)
破壁人口胡出来的,感觉很是奇妙。
由于四个状态没法直接 2SAT,可以每个状态拆成两部分:
于是就可以建出一个边数 $O(n^2)$ 的图,然后依次最大化每一个矩形的答案,再 2SAT 判断是否合法,复杂度 $O(n^3)$。