Luogu P9509『STA - R3』Aulvwc 题解

简化一下题意,对于原集合 $S$,就是找原集合的两个子集 $S_1,S_2$,使得 $S_1 \cup S_2 = S,S_1 \cap S_2 = \empty$,$S_1,S_2$ 的平均数等于 $S$ 的平均数

注意到特殊性质中的 $\sum a_i = 0$,这个很好写,分成一个正集合和一个负集合,每个跑一个可行性背包 $f,g$,$f_i,g_i$ 表示 $S_1,S_2$ 中能否存在 $i$,如果 $f_i = g_i,i \in [1,$ $\sum a_j \over 2$ $(a_j > 0))$,那么$S_1,S_2$ 的平均数等于 $S$ 的平均数

考虑把整体情况转化,把 $a_i$ 减掉平均数,那么 $\sum a_i = 0$,这样就成了特殊性质。按照特殊性质的方法处理就行。

但时间复杂度是 $O(n^2k)$($k$ 是值域,题中为 $5\times 10^3$),大约是个 $5 \times 10^9$,会炸,考虑用 bitset 优化可行性背包,f = f|=(f<<a_i),时间复杂度优化到了 $10^8$ 左右,可以通过。

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
#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e6+10;
int n,a[N];
bitset<2500000> fp,fn;
void solve(){
int sum=0,agv=0;
fp=fn=0;
vector<int> nn,pp;
cin>>n;
for(int i=1;i<=n;i++) cin>>a[i],agv+=a[i];
if(agv%n!=0){
puts("No");
return;
}
agv/=n;
for(int i=1;i<=n;i++){
a[i]-=agv;
if(a[i]==0){
puts("Yes");
return;
}
(a[i]>0?pp:nn).push_back(abs(a[i]));
sum+=(a[i]>0?a[i]:0);
}
fp[0]=fn[0]=1;
for(int x:pp) fp|=(fp<<x);
for(int x:nn) fn|=(fn<<x);
fp[sum]=fn[sum]=fp[0]=fn[0]=0;
puts(((fp&fn)!=0?"Yes":"No"));
}
signed main(){
cin.tie(0)->ios::sync_with_stdio(0);
int q;cin>>q;
while(q--) solve();
return 0;
}

共发表 27 篇Blog · 总计 22.4k 字
© 2025 AirTouch 使用 Stellar 创建
萌ICP备20250662号 雾备 88666688号 网ICP备20258888号
本站总访问量次 本站访客数人次 本文总阅读量