[SCOI2008]城堡

最大值最小,显然二分答案,但考虑二分后如何 check

\(n\) 个点 \(n\) 条边,显然这是一个基环树森林。对于基环树,常用的套路是拆环为链,枚举删去哪条边。但这题是基环树森林,拆环为链的复杂太高,考虑将环和树分开处理。

树上是一个很典型的 dp,和将军令一样(不了解的可以观看我的相关博客),注意处理下原来关键点的影响即可。

环上的部分要分成两种情况:

  • 第一种是环上没有关键点。需要先处理该点是否被其它点子树内的关键点覆盖。令 \(len\) 表示处理完子树内的点后还可以影响环上多少长度,若 \(len\times 2>\) 环的长度,则整个环只需要新添一个点就可以全部被覆盖。否则会变成一个经典贪心问题,在环上有一些区间,每个区间表示表示关键点被放置在该区间时,某个点才会被覆盖,求最少要有多少区间才能覆盖这个环。
  • 环上有关键点,其实也差不多,只是最后求最少区间覆盖环时有了一个起点。
点击查看代码
#include<bits/stdc++.h>
#define ull unsigned long long
#define ll long long
#define pdi pair<double,int>
#define pii pair<int,int>
#define pb push_back
#define mp make_pair
#define eps 1e-9
using namespace std;
namespace IO{
template<typename T>
inline void read(T &x){
x=0;
int f=1;
char ch=getchar();
while(ch>'9'||ch<'0'){
if(ch=='-'){
f=-1;
}
ch=getchar();
}
while(ch>='0'&&ch<='9'){
x=x*10+(ch-'0');
ch=getchar();
}
x=(f==1?x:-x);
}
template<typename T>
inline void write(T x){
if(x<0){
putchar('-');
x=-x;
}
if(x>=10){
write(x/10);
}
putchar(x%10+'0');
}
template<typename T>
inline void write_endl(T x){
write(x);
putchar('\n');
}
template<typename T>
inline void write_space(T x){
write(x);
putchar(' ');
}
}
using namespace IO;
const int N=200,inf=1e9;
int n,m,K,d[N][N],v[N],w[N],flag[N],vis[N],f[N],g[N];
int idx,Time;
vector<pii>e[N];
stack<int>stk;
struct node{
int l,r;
};
struct tree{
int circle[N],cnt,rank[N],pos,dis[N],s[N],in_cir[N],sum,k,tot,R[N],nxt[N];
node seg[N];
void ins(int x){
circle[++cnt]=x;
rank[x]=cnt;
in_cir[x]=1;
}
void work(){
for(int i=1;i<=cnt;i++){
if(flag[circle[i]]){
pos=i;
break;
}
}
for(int i=1;i<=cnt;i++){
dis[i]=dis[i+cnt]=d[circle[i]][circle[i%cnt+1]];
}
for(int i=1;i<=cnt*2;i++){
s[i+1]=s[i]+dis[i];
}
}
void dfs(int u,int fa){
f[u]=0;
g[u]=inf;
for(auto x:e[u]){
int v=x.first,w=x.second;
if(in_cir[v]||v==fa){
continue;
}
dfs(v,u);
f[u]=max(f[u],f[v]+w);
g[u]=min(g[u],g[v]+w);
}
if(flag[u]){
g[u]=0;
}
if(f[u]+g[u]<=k){
f[u]=-inf;
}
if(f[u]+d[u][fa]>k){
f[u]=-inf;
g[u]=0;
sum++;
}
}
int solve(){
// cerr<<k<<endl;
tot=sum=0;
for(int i=1;i<=cnt;i++){
dfs(circle[i],0);
}
int fl=0;
for(int i=1;i<=cnt*2;i++){
R[i]=inf;
}
for(int i=1;i<=cnt;i++){
bool flg=0;
for(int j=1;j<=cnt;j++){
int Dis=abs(s[i]-s[j]);
Dis=min(Dis,s[cnt+1]-Dis);
if(f[circle[i]]+g[circle[j]]+Dis<=k){
flg=1;
break;
}
}
if(flg){
continue;
}
int len=k-f[circle[i]];
if(len*2>=s[cnt+1]){
if(!pos){
fl=1;
}
continue;
}
++tot;
if(s[i]+dis[cnt]>len){
seg[tot].l=1,seg[tot].r=2*cnt;
for(int j=i;j;j--){
if(s[i]-s[j]>len){
seg[tot].l=j+1;
break;
}
}
for(int j=i+1;j<=cnt*2;j++){
if(s[j]-s[i]>len){
seg[tot].r=j-1;
break;
}
}
seg[tot+1].l=seg[tot].l+cnt;
seg[tot+1].r=seg[tot].r+cnt;
++tot;
}
else{
seg[tot].l=1,seg[tot].r=2*cnt;
for(int j=i+cnt;j;j--){
if(s[i+cnt]-s[j]>len){
seg[tot].l=j+1;
break;
}
}
for(int j=i+cnt;j<=2*cnt;j++){
if(s[j]-s[i+cnt]>len){
seg[tot].r=j-1;
break;
}
}
}
}
// cerr<<tot<<' '<<fl<<endl;
if(!tot){
return sum+fl;
}
for(int i=1;i<=tot;i++){
R[seg[i].l]=min(R[seg[i].l],seg[i].r);
}
int c=0,mn=inf;
for(int i=2*cnt;i;i--){
nxt[i]=mn;
mn=min(mn,R[i]);
}
if(pos){
int x=pos;
while(nxt[x]<pos+cnt){
c++;
x=nxt[x];
}
}
else{
c=inf;
for(int i=1;i<=cnt;i++){
int x=i,S=1;
while(nxt[x]<i+cnt){
S++;
x=nxt[x];
}
c=min(c,S);
}
}
c=max(c,fl);
return sum+c;
}
}tr[N];
void visit(int u,int fa){
vis[u]=Time;
for(auto x:e[u]){
int v=x.first;
if(vis[v]==Time){
continue;
}
visit(v,u);
}
}
bool dfs(int u,int fa){
stk.push(u);
vis[u]=Time;
bool First=1;
for(auto x:e[u]){
int v=x.first;
if(vis[v]==Time){
if(v==fa){
if(First){
First=0;
continue;
}
}
while(stk.top()!=v){
tr[idx].ins(stk.top());
stk.pop();
}
tr[idx].ins(stk.top());
stk.pop();
return 1;
}
if(dfs(v,u)){
return 1;
}
}
stk.pop();
return 0;
}
signed main(){
#ifndef ONLINE_JUDGE
freopen("1.in","r",stdin);
freopen("1.out","w",stdout);
#endif
read(n),read(m),read(K);
for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
d[i][j]=inf;
}
}
for(int i=1;i<=n;i++){
read(v[i]);
v[i]++;
}
for(int i=1;i<=n;i++){
read(w[i]);
e[i].pb(mp(v[i],w[i]));
e[v[i]].pb(mp(i,w[i]));
d[i][v[i]]=d[v[i]][i]=min(d[i][v[i]],w[i]);
}
for(int i=1;i<=m;i++){
int x;
read(x);
flag[x+1]=1;
}
for(int i=1;i<=n;i++){
if(!vis[i]){
Time++;
idx++;
visit(i,0);
Time++;
dfs(i,0);
tr[idx].work();
}
}
int l=0,r=inf,ans=inf;
while(l<=r){
int mid=(l+r)>>1,x=0;
// cerr<<mid<<' ';
for(int i=1;i<=idx;i++){
tr[i].k=mid;
x+=tr[i].solve();
}
if(x>K){
l=mid+1;
}
else{
r=mid-1;
ans=mid;
}
}
write_endl(ans);
return 0;
}

[[SCOI2008]城堡] 解题报告的更多相关文章

  1. 夏令营提高班上午上机测试 Day 4 解题报告

    我要是没记错的话,今天的题难度算挺适中的. *标程来自高天宇哥哥 T1:小G的字符串 题目描述 有一天,小 L 给小 G 出了这样一道题:生成一个长度为 n 的.全由小写英文字母构成的字符串,只能使用 ...

  2. CH Round #56 - 国庆节欢乐赛解题报告

    最近CH上的比赛很多,在此会全部写出解题报告,与大家交流一下解题方法与技巧. T1 魔幻森林 描述 Cortana来到了一片魔幻森林,这片森林可以被视作一个N*M的矩阵,矩阵中的每个位置上都长着一棵树 ...

  3. 二模13day1解题报告

    二模13day1解题报告 T1.发射站(station) N个发射站,每个发射站有高度hi,发射信号强度vi,每个发射站的信号只会被左和右第一个比他高的收到.现在求收到信号最强的发射站. 我用了时间复 ...

  4. BZOJ 1051 最受欢迎的牛 解题报告

    题目直接摆在这里! 1051: [HAOI2006]受欢迎的牛 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4438  Solved: 2353[S ...

  5. 习题:codevs 2822 爱在心中 解题报告

    这次的解题报告是有关tarjan算法的一道思维量比较大的题目(真的是原创文章,希望管理员不要再把文章移出首页). 这道题蒟蒻以前做过,但是今天由于要复习tarjan算法,于是就看到codevs分类强联 ...

  6. 习题:codevs 1035 火车停留解题报告

    本蒟蒻又来写解题报告了.这次的题目是codevs 1035 火车停留. 题目大意就是给m个火车的到达时间.停留时间和车载货物的价值,车站有n个车道,而火车停留一次车站就会从车载货物价值中获得1%的利润 ...

  7. 习题: codevs 2492 上帝造题的七分钟2 解题报告

    这道题是受到大犇MagHSK的启发我才得以想出来的,蒟蒻觉得自己的代码跟MagHSK大犇的代码完全比不上,所以这里蒟蒻就套用了MagHSK大犇的代码(大家可以关注下我的博客,友情链接就是大犇MagHS ...

  8. 习题:codevs 1519 过路费 解题报告

    今天拿了这道题目练练手,感觉自己代码能力又增强了不少: 我的思路跟别人可能不一样. 首先我们很容易就能看出,我们需要的边就是最小生成树算法kruskal算法求出来的边,其余的边都可以删掉,于是就有了这 ...

  9. NOIP2016提高组解题报告

    NOIP2016提高组解题报告 更正:NOIP day1 T2天天爱跑步 解题思路见代码. NOIP2016代码整合

  10. LeetCode 解题报告索引

    最近在准备找工作的算法题,刷刷LeetCode,以下是我的解题报告索引,每一题几乎都有详细的说明,供各位码农参考.根据我自己做的进度持续更新中......                        ...

随机推荐

  1. CH573 CH582 CH579蓝牙从机(peripheral)例程讲解四(蓝牙动态广播)

    动态广播有两种实现方式: 1.关闭广播,更改广播包数据,等待关闭上报状态后,开启广播. uint8_t initial_advertising_enable = FALSE; GAPRole_SetP ...

  2. Leecode 21.合并两个有序链表(Java 迭代、递归两种方法)

      想法: 1.迭代 设两个指针pa和pb,不断移动pa和pb,并进行比较,则将较小元素接到新链表,该过程直至pa或pb为null,之后将未空的接到已空之后,得到升序链表   1 //官方: 2 cl ...

  3. python实例1(石头 剪刀 布)

    #random   .randint       模块导入 import random #定义一个用户需要输入的数据内容入口 user = int(input("请输入(石头1,剪刀2,布3 ...

  4. adb shell monkey

    monkey 在模拟器或真机里在,.它向系统发送伪随机的用户事件流(如按键输入.触摸屏输入.手势输入等),打开app 查看包名 adb shell dumpsys window windows | f ...

  5. SpringBoot笔记--自动配置(高级内容)(中集)

    @Enable*注解 使用该注解,需要导入相应的依赖坐标,其中的groupId标签里面写入Bean的Java文件所在的包的路径下面 spring-enable-other 还需要在SpringBoot ...

  6. 文件的上传&预览&下载学习(三)

    0.参考博客 https://www.pianshen.com/article/18961690151/ (逻辑流程图讲得很清楚) https://www.cnblogs.com/xiahj/p/vu ...

  7. mybatis源码-注解sql

    Mybatis-注解sql Demo 主启动类 public class MybatisHelloWorld { public static void main(String[] args) thro ...

  8. Why WebRTC|“浅入深出”的工作原理详解

    前言 近几年实时音视频通信应用呈现出了大爆发的趋势.在这些实时通信技术的背后,有一项不得不提的技术--WebRTC. 今年 1 月,WebRTC 被 W3C 和 IETF 发布为正式标准.据调研机构 ...

  9. ARP协议:网络世界的临门一脚

    大家好,我是风筝. 各位同学肯定见过关于网络的面试题,什么TCP协议和UDP的区别啦,IP协议工作在哪层啊等等,这都是网络中定义的各种协议.这些标准化的协议就是网络分层模型标准化的核心部分.要想搞懂网 ...

  10. 高性能 Kafka队列的原理

    一.原理简述 [1]Producer将消息进行分组分别发送到对应 Leader节点:[2]Leader将消息写入本地 log:[3]Followers从 Leader pull最新消息,写入 log后 ...