99 lines
2.3 KiB
C++
99 lines
2.3 KiB
C++
|
#include<bits/stdc++.h>
|
|||
|
#define MAXN 200005
|
|||
|
#define P int(1e9+7)
|
|||
|
using namespace std;
|
|||
|
typedef long long ll;
|
|||
|
/*
|
|||
|
U111091 区间2段覆盖
|
|||
|
先看k==1的情况:
|
|||
|
观察1:
|
|||
|
存在最优区间以某个村庄作为终点
|
|||
|
那么做法就是枚举终点,维护一个指针指向起点,计算答案即可,复杂度 。
|
|||
|
当k==2时,需要考虑两种情况:
|
|||
|
1. 第一个区间终点和第二个区间起点在同一条道路内
|
|||
|
2. 否则,存在一个分界村庄i,2条道路分别在i之前和i之后
|
|||
|
观察2:
|
|||
|
若2个选定区间的端点在同一条道路内,则存在最优方案使得它们首尾相连
|
|||
|
调整法可以证明,所以case1,就当作 做;
|
|||
|
对于case2,分别dp之后维护前缀/后缀max,枚举分界点 即可。
|
|||
|
总复杂度线性
|
|||
|
|
|||
|
*/
|
|||
|
int T,N,K,q;
|
|||
|
int x[MAXN], sum = 0;
|
|||
|
int g[MAXN], pre[MAXN], suf[MAXN];
|
|||
|
char s[MAXN];
|
|||
|
int solve1(int q)//解决优化一段长度为q的土路的情况
|
|||
|
{
|
|||
|
int k = 1;//一段路的开始位置
|
|||
|
int ans = 0;//能优化的最长的土路的长度
|
|||
|
int ans1;
|
|||
|
for(int i=2;i<=N;i++)
|
|||
|
{
|
|||
|
while(x[i]-x[k]>q)
|
|||
|
k++;
|
|||
|
ans1 = g[i] - g[k]; //第k个村庄到第i个村庄之间土路的长度
|
|||
|
if(s[k]=='1')
|
|||
|
ans1 += q - (x[i]-x[k]);
|
|||
|
ans = max(ans, ans1);
|
|||
|
}
|
|||
|
//cerr<<"ans1 = "<<ans1<<endl;
|
|||
|
return ans;
|
|||
|
}
|
|||
|
int solve2(int q)//计算找两段长度为q的土路,最多能改变多少变成高速公路
|
|||
|
{
|
|||
|
memset(pre, 0, sizeof(pre));
|
|||
|
memset(suf, 0, sizeof(suf));
|
|||
|
|
|||
|
int ans1;
|
|||
|
int k = 1;//一段路的开始位置
|
|||
|
for(int i=2;i<=N;i++){
|
|||
|
while(x[i]-x[k]>q) ++k;
|
|||
|
ans1 = g[i] - g[k];
|
|||
|
if(s[k]=='1')
|
|||
|
ans1 += q - (x[i]-x[k]);
|
|||
|
pre[i] = max(pre[i-1], ans1);
|
|||
|
}
|
|||
|
k = N;
|
|||
|
for(int i=N-1;i>=1;i--){
|
|||
|
while(x[k]-x[i]>q)
|
|||
|
--k;
|
|||
|
ans1 = g[k] - g[i];
|
|||
|
if(s[k+1]=='1')
|
|||
|
ans1 += q - (x[k]-x[i]);
|
|||
|
suf[i] = max(suf[i+1], ans1);
|
|||
|
}
|
|||
|
int ans = 0;
|
|||
|
for(int i=2;i<=N;i++){
|
|||
|
ans = max(ans, pre[i] + suf[i]);
|
|||
|
}
|
|||
|
return ans;
|
|||
|
}
|
|||
|
int main(){
|
|||
|
cin>>T;
|
|||
|
while(T--){
|
|||
|
cin>>N>>K>>q;
|
|||
|
for(int i=1;i<=N;i++)
|
|||
|
cin>>x[i];
|
|||
|
cin>>s+2;
|
|||
|
s[1] = '0';
|
|||
|
sum = 0;//土路总长度
|
|||
|
for(int i=2;i<=N;i++){
|
|||
|
g[i] = g[i-1];//前i个村庄之间土路的总长度
|
|||
|
if(s[i]=='1')//i-1到i村庄之间的路是土路
|
|||
|
{
|
|||
|
g[i] += x[i] - x[i-1];
|
|||
|
sum += x[i] - x[i-1];
|
|||
|
}
|
|||
|
}
|
|||
|
int ans;
|
|||
|
if(K==1)
|
|||
|
ans = sum - solve1(q);
|
|||
|
else
|
|||
|
ans = sum - max(solve1(2*q), solve2(q));
|
|||
|
cout<<ans<<'\n';
|
|||
|
}
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|