令$(S_{a},S_{b})$表示$a_{i}\in S_{a}$且$b_{i}\in S_{b}$的i个数,那么答案相当于$S(0,1)+S(1,0)=S(0,1)+S(\{0,1\},0)-S(0,0)$,容易发现$S(\{0,1\},0)=\sum_{i=1}^{n}[b_{i}==0]$,那么相当于最小化$S(0,1)-S(0,0)$,因此答案与1的位置无关
然后dp,用$f[i][j]$表示前i个点最小的$S(0,1)-S(0,0)$且$\forall \min(i,j)\le k\le n,a_{k}=[k\le j]$,考虑转移:
1.如果$i\le j$,那么$f[i][j]=\min(f[i][j],f[i-1][j])$;如果$j<i$,那么$f[i][j]=\min(f[i][j],f[i-1][j]+2b_{i}-1)$
2.对于操作区间$[l,r]$,如果$l=i$,那么$f[i][r]=\min(f[i][r],\min_{i-1\le j\le r}(f[i-1][j]))$(注意滚动后要从大到小枚举r)
容易发现以下两种转移都是由$f[i-1]$转移到$f[i]$,用线段树维护,支持区间修改,区间查询最小值,单点取min即可(初始值应该赋为无穷大),最终答案即为$\min(f[n][i])+\sum_{i=1}^{n}[b_{i}==0]$
1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 200005
4 #define L (k<<1)
5 #define R (L+1)
6 #define mid (l+r>>1)
7 vector<int>v[N];
8 int n,m,x,y,ans,a[N],tag[N<<2],f[N<<2];
9 void upd(int k,int x){
10 tag[k]+=x;
11 f[k]+=x;
12 }
13 void down(int k){
14 if (tag[k]){
15 upd(L,tag[k]);
16 upd(R,tag[k]);
17 tag[k]=0;
18 }
19 }
20 void update(int k,int l,int r,int x,int y){
21 if (l==r){
22 f[k]=min(f[k],y);
23 return;
24 }
25 down(k);
26 if (x<=mid)update(L,l,mid,x,y);
27 else update(R,mid+1,r,x,y);
28 f[k]=min(f[L],f[R]);
29 }
30 void update(int k,int l,int r,int x,int y,int z){
31 if ((l>y)||(x>r))return;
32 if ((x<=l)&&(r<=y)){
33 upd(k,z);
34 return;
35 }
36 down(k);
37 update(L,l,mid,x,y,z);
38 update(R,mid+1,r,x,y,z);
39 f[k]=min(f[L],f[R]);
40 }
41 int query(int k,int l,int r,int x,int y){
42 if ((l>y)||(x>r))return 0x3f3f3f3f;
43 if ((x<=l)&&(r<=y))return f[k];
44 return min(query(L,l,mid,x,y),query(R,mid+1,r,x,y));
45 }
46 int main(){
47 scanf("%d",&n);
48 for(int i=1;i<=n;i++){
49 scanf("%d",&a[i]);
50 ans+=(a[i]^1);
51 }
52 scanf("%d",&m);
53 for(int i=1;i<=m;i++){
54 scanf("%d%d",&x,&y);
55 v[x].push_back(y);
56 }
57 for(int i=1;i<=n;i++)sort(v[i].begin(),v[i].end());
58 memset(f,0x3f,sizeof(f));
59 update(1,0,n,0,0);
60 for(int i=1;i<=n;i++){
61 for(int j=(int)v[i].size()-1;j>=0;j--)update(1,0,n,v[i][j],query(1,0,n,0,v[i][j]));
62 update(1,0,n,0,i-1,2*a[i]-1);
63 }
64 printf("%d",f[1]+ans);
65 }