题意

有一棵 \(n\) 个点的树和 \(m\) 个人,第 \(i\) 个人从 \(u_i\) 走到 \(v_i\)

现在要发宠物,要求一个人要么他自己发到宠物,要么他走的路径上的都有宠物。

求最小代价,并输出任意方案。

\(n,m \leq 20000\)

传送门

思路

对每个人和每条树边都建一个点。

源点向每个人连容量 \(1\) 的边,每条树边向汇点连容量 \(1\) 的边。每个人向他要走到的所有边连容量 \(+\infty\) 的边。

给人发就是割掉人与源的边,放边上就是割树边与汇的边,人与树边间的边不能割。当源汇不连通的时候,就是满足题意的,题目转化为最小割。

问题出在连边上,这是 \(nm\) 的,考虑如何优化连边。线段树?树上问题有点麻烦,想不到倍增。

对于一段相邻的\(2^i\)个点,都建一个点与它们所有相连,其实又有点像线段树,然后就会对于每条链得到一棵类似线段树的东西?大区间连向小区间,看起来就是。



然后对每个人,将他的路径在 LCA 处分成两条路径,这两条路径分别向对应区间覆盖。

那么我们得到的新图有 \(n\log n+m\) 个点和 \(n\log n+4m\) 条边,可以通过。

这里还要输出方案。

我们知道,最小割中的边一定满流。(全局最大流,分成两个集合,其间的边一定是满流的,而也就是最小割)因此我们从源点开始 dfs,只走没满流的边,并标记被 dfs 到的点。则图被分成两部分,一部分被访问过,一部分没被访问过。其中间那些边就是一个最小割。

然后,如果一个人代表的点没被访问过,则说明他所属那条边被割了。如果一条树边代表的点被访问过,则说明它所属那条边被割了。记录标号输出

#include <bits/stdc++.h>
using std::queue;
const int W=14,N=20005*(W+1),M=N+N*4;
int to[M<<1],w[M<<1],Next[M<<1],edge,n,x,y,f[20005][W+1],idn[N],deep[N],last[N],b[20005][W+1];
int s,t,m,cnt,ans,tag[N],cur[N];
queue <int> q;
void add(int x,int y,int z){
to[++edge]=y;
Next[edge]=last[x];
last[x]=edge;
w[edge]=z;
}
void dfs(int x,int fa){
f[x][0]=fa,deep[x]=deep[fa]+1;
for (int i=last[x];i;i=Next[i])
if (to[i]!=fa) {
idn[to[i]]=(i+1)/2;
dfs(to[i],x);
}
}
void Add(int x,int y,int w){
add(x,y,w);
add(y,x,0);
}
int lca(int x,int y){
if (deep[x]<deep[y]) std::swap(x,y);
for (int i=W;i>=0;i--)
if (deep[f[x][i]]>=deep[y]) x=f[x][i];
if (x==y) return x;
for (int i=W;i>=0;i--)
if (f[x][i]!=f[y][i]) x=f[x][i],y=f[y][i];
return f[x][0];
}
void addedge(int x,int y,int id){
for (int i=W;i>=0;i--)
if (deep[f[x][i]]>=deep[y]){
Add(id,b[x][i],n+1);
x=f[x][i];
}
}
bool bfs(){
for (int i=0;i<=t;i++) cur[i]=last[i],deep[i]=0;
deep[s]=1;
q.push(s);
while (!q.empty()){
int x=q.front();
q.pop();
for (int i=last[x];i;i=Next[i])
if (w[i] && !deep[to[i]]){
deep[to[i]]=deep[x]+1;
q.push(to[i]);
}
}
return (deep[t]);
}
int c(int x) { return x&1?x+1:x-1; }
int dfs(int x){
if (x==t) return 1;
for (int i=cur[x];i;i=Next[i]){
cur[x]=i;
int u=to[i];
if (deep[u]>deep[x] && w[i]){
int di=dfs(to[i]);
if (di){
w[i]--;
w[c(i)]++;
return 1;
}
}
}
return 0;
}
void DFS(int x){
tag[x]=1;
for (int i=last[x];i;i=Next[i])
if (w[i] && !tag[to[i]]) DFS(to[i]);
}
int main(){
scanf("%d%d",&n,&m);
for (int i=1;i<n;i++){
scanf("%d%d",&x,&y);
add(x,y,0),add(y,x,0);
}
s=0;
dfs(1,0);
edge=0;memset(last,0,sizeof(last));
cnt=n;
for (int i=2;i<=n;i++) b[i][0]=i;
for (int i=1;i<=W;i++)
for (int j=1;j<=n;j++)
if (f[f[j][i-1]][i-1]){
f[j][i]=f[f[j][i-1]][i-1];
Add(++cnt,b[j][i-1],n+1);
Add(cnt,b[f[j][i-1]][i-1],n+1);
b[j][i]=cnt;
}
for (int i=1;i<=m;i++){
scanf("%d%d",&x,&y);
Add(s,++cnt,1);
idn[cnt]=i;
int l=lca(x,y);
addedge(x,l,cnt),addedge(y,l,cnt);
}
t=cnt+1;
for (int i=2;i<=n;i++) Add(i,t,1);
while (bfs())
while (dfs(s)==1) ans++;
printf("%d\n",ans);
DFS(s);
ans=0;
for (int i=last[s];i;i=Next[i])
if (tag[to[i]]!=tag[s]) ans++;
printf("%d ",ans);
for (int i=last[s];i;i=Next[i])
if (tag[to[i]]!=tag[s]) printf("%d ",idn[to[i]]);
puts("");
ans=0;
for (int i=last[t];i;i=Next[i])
if (tag[to[i]]!=tag[t]) ans++;
printf("%d ",ans);
for (int i=last[t];i;i=Next[i])
if (tag[to[i]]!=tag[t]) printf("%d ",idn[to[i]]);
return 0;
}

后记

追随神仙的脚步

抄袭于此Mrsrz

CF786E ALT的更多相关文章

  1. cf786E ALT (最小割+倍增优化建图)

    如果把“我全都要”看作是我全不要的话,就可以用最小割解决啦 源点S,汇点T 我们试图让每个市民作为一个等待被割断的路径 把狗狗给市民:建边(S,i,1),其中i是市民 把狗狗给守卫:建边(j,T,1) ...

  2. 一句话题解&&总结

    CF79D Password: 差分.两点取反,本质是匹配!最短路+状压DP 取反是套路,匹配是发现可以把操作进行目的化和阶段化,从而第二次转化问题. 且匹配不会影响别的位置答案 sequence 计 ...

  3. [No000093]按住Alt 再按数字键敲出任意汉字和字符!

    1.在notepad里,(中文系统下) 按住Alt 然后按52946最后放开Alt 按住Alt 然后按45230最后放开Alt 按住Alt 然后按50403最后放开Alt 你会看到"我爱你& ...

  4. [No00008B]远程桌面发送“Ctrl+Alt+Delete”组合键调用任务管理器

    向远程桌面发送"Ctrl+Alt+Delete"组合键的两种方法 1.在本地按下Ctrl+Alt+End,可以成功发送"Ctrl+Alt+Delete"组合键! ...

  5. 平常看到的Alt+xx 快捷键用法

    1. 先按Alt, 哪一个菜单对应的字符是有划线的. 2. 输入对应的字符打开相应的菜单, 3 再输入相应的字符打开子菜单

  6. windows 中去除Ctrl+Alt+Del才能登录

    安装windows 7后登录的时候有一样很麻烦的步骤是需要先按Ctrl+Alt+Del,才能输入用户密码进行登录.这里笔者介绍一下如何取消这个东西. 点击“开始菜单”,点击“控制面板”. [管理工具] ...

  7. TSql 巧用Alt 键

    1,查看表的信息 在TSql 编辑器中,选中一个表,如图 点击Alt+F1,就可以查看表的属性定义 2,使用alt批量插入逗号 在Tsql中使用 in 子句,在(value_List)列表中,经常有很 ...

  8. title与alt的区别

    html中的title属性和alt属性让人有些混淆. 以前不知道有title这个属性,第一次用到它时,就和alt产生了混淆.一位朋友告诉我说,alt是图片img标签里用的,title是超链接里用的, ...

  9. UML序列图总结(Loop、Opt、Par和Alt)

    序列图主要用于展示对象之间交互的顺序. 序列图将交互关系表示为一个二维图.纵向是时间轴,时间沿竖线向下延伸.横向轴代表了在协作中各独立对象的类元角色.类元角色用生命线表示.当对象存在时,角色用一条虚线 ...

随机推荐

  1. Neo4J之标签类型

    Neo4J的标签可以理解一个类,在创建一个节点时可以设置一个或多个标签: 1. 标签名为中文(可以) CRATE(节点名:标签1:标签2{属性1:34} 创建了一个节点名为“节点名”的节点(不可以用节 ...

  2. Java内存模型之总结

    经过四篇博客阐述,我相信各位对Java内存模型有了最基本认识了,下面LZ就做一个比较简单的总结. 总结 JMM规定了线程的工作内存和主内存的交互关系,以及线程之间的可见性和程序的执行顺序.一方面,要为 ...

  3. 【转载】如何自己DIY组装一台台式电脑

    针对很多懂计算机的人员来说,有时候都希望自己DIY组装一台台式机,来达到自己的个性化要求以及省钱.其实自己DIY组装一台电脑也很简单,将相应的CPU处理器.主板.内存条.硬盘.固态硬盘.电脑机箱.屏幕 ...

  4. 【转载】C#通过StartWith和EndWith方法判断字符串是否以特定字符开始或者结束

    C#开发过程中针对字符串String类型的操作是常见操作,有时候业务需要判断某个字符串是否以特定字符开头或者特定字符结束,此时就可使用StartsWith方法来判断目标字符串是否以特定字符串开头,通过 ...

  5. window.postMessage()实现(iframe嵌套页面)跨域消息传递

    window.postMessage()方法可以安全地实现Window对象之间的跨域通信.例如,在页面和嵌入其中的iframe之间. 不同页面上的脚本允许彼此访问,当且仅当它们源自的页面共享相同的协议 ...

  6. 解决JAVA连接Sybase数据库查询数据乱码的问题

    连接字符串加上charset=eucgb&jconnect_version=0例如:jdbc:sybase:Tds:server:port/database?charset=eucgb& ...

  7. UI5-技术篇-Hybrid App-1-Barcode扫描

    参考资料: https://www.w3cschool.cn/cordova/cordova_overview.html https://blogs.sap.com/2017/01/03/sapui5 ...

  8. idea 实用插件

    尊重劳动成果,本插件的整理原文出自:https://blog.csdn.net/weixin_41846320/article/details/82697818,感谢老铁的辛苦原创. 插件安装方式: ...

  9. k8s的Pod状态和生命周期管理

    Pod状态和生命周期管理   一.什么是Pod? 二.Pod中如何管理多个容器? 三.使用Pod 四.Pod的持久性和终止 五.Pause容器 六.init容器 七.Pod的生命周期 (1)Pod p ...

  10. 中文日历Calendar

    一.层次结构 Object<-----Calendar<-----EastAsianLunisolarCalendar<-----ChineseLunisolarCalendar(农 ...