[NOIP2015]运输计划

链接

luogu

好久没写博客了,水一篇波。

思路1 暴力数据结构

枚举最长链的边,删除后代价为(最长链-边权,不经过这条边的链)的最大值。

不经过某条边的最大值要用线段树维护补集。

复杂度\(O(nlog^2n)\)

思路2 二分树上差分

二分答案,删除的边为\(>mid\)的链的交集。

用树上查分维护交集。

最后在交集中找个最大的边删除就好了。

复杂度\(O(nlogn)\)

总的

思路2,复杂度小,好写,简单,细节少。

思路1,复杂度高,码量大,细节多,并不是呢么好想。

所以并不是很建议写第一种。

但他靠着树剖的小常数跑过了树上差分(至少我是这样,1.9s->1.2s)。

代码1

#include <bits/stdc++.h>
using namespace std;
const int _=5e5+7;
int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f=-1;
for(;s>='0'&&s<='9';s=getchar()) x=x*10+s-'0';
return x*f;
}
int n,m,ans,u[_],v[_],dsr[_],w[_];
struct node {
int v,nxt,q,w;
}e[_<<1];
int head[_],tot;
void add(int u,int v,int q) {
e[++tot].v=v;
e[tot].q=q;
e[tot].nxt=head[u];
head[u]=tot;
}
int f[_],dep[_],siz[_],son[_],top[_],idx[_],dis[_],nb[_],cnt;
void dfs1(int u,int fa) {
dep[u]=dep[fa]+1;
f[u]=fa;
siz[u]=1;
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(v==fa) continue;
w[v]=e[i].q;
dis[v]=dis[u]+e[i].q;
dfs1(v,u);
siz[u]+=siz[v];
if(siz[son[u]]<siz[v]) son[u]=v;
}
}
void dfs2(int u,int topf) {
top[u]=topf;
idx[u]=++cnt;
nb[cnt]=u;
if(!son[u]) return;
dfs2(son[u],topf);
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(!idx[v]) dfs2(v,v);
}
}
namespace seg_tree {
#define ls rt<<1
#define rs rt<<1|1
int ma[_<<2],lazy[_];
void tag(int rt,int val) {ma[rt]=max(ma[rt],val);lazy[rt]=max(lazy[rt],val);}
void pushdown(int rt) {tag(ls,lazy[rt]);tag(rs,lazy[rt]);}
void modify(int L,int R,int val,int l,int r,int rt) {
if(L>R) return;
if(L<=l&&r<=R) return tag(rt,val);
int mid=(l+r)>>1;
pushdown(rt);
if(L<=mid) modify(L,R,val,l,mid,ls);
if(R>mid) modify(L,R,val,mid+1,r,rs);
ma[rt]=max(ma[ls],ma[rs]);
}
void dfs(int l,int r,int rt) {
if(l==r) return void(dsr[nb[l]]=ma[rt]);
int mid=(l+r)>>1;
pushdown(rt);
if(l<=mid) dfs(l,mid,ls);
if(r>mid) dfs(mid+1,r,rs);
}
}
int lca(int x,int y) {
while(top[x]!=top[y]) {
if(dep[top[x]]<dep[top[y]]) swap(x,y);
x=f[top[x]];
} return dep[x]<dep[y]?x:y;
}
void QQ(int x,int y,int val) {
int las=n;
while(top[x]!=top[y]) {
if(dep[top[x]]<dep[top[y]]) swap(x,y);
seg_tree::modify(idx[x]+1,las,val,1,n,1);
las=idx[top[x]]-1;
x=f[top[x]];
}
if(dep[x]<dep[y]) swap(x,y);
seg_tree::modify(idx[x]+1,las,val,1,n,1);
seg_tree::modify(1,idx[y],val,1,n,1);
}
int main() {
n=read(),m=read();
for(int i=1,x,y,q;i<n;++i) {
x=read(),y=read(),q=read();
add(x,y,q),add(y,x,q);
}
for(int i=1;i<=m;++i) {
u[i]=read(),v[i]=read();
if(u[i]==v[i]) m--,i--;
}
dfs1(1,0),dfs2(1,1);
int id=0,ma=0;
for(int i=1;i<=m;++i) {
int val=dis[u[i]]+dis[v[i]]-2*dis[lca(u[i],v[i])];
QQ(u[i],v[i],val);
if(ma<val) id=i,ma=val;
}
if(!id) return printf("0\n"),0;
seg_tree::dfs(1,n,1);
int x=u[id],y=v[id],ans=0x3f3f3f3f;
while(x!=y) {
if(dep[x]<dep[y]) swap(x,y);
ans=min(ans,max(dsr[x],ma-w[x]));
x=f[x];
}
cout<<ans<<"\n";
return 0;
}

代码2

#include <bits/stdc++.h>
using namespace std;
const int _=5e5+7;
int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f=-1;
for(;s>='0'&&s<='9';s=getchar()) x=x*10+s-'0';
return x*f;
}
int n,m,ans,u[_],v[_],LCA[_],len[_],the_biggest,w[_];
struct node {
int v,nxt,q;
}e[_<<1];
int head[_],tot;
void add(int u,int v,int q) {
e[++tot].v=v;
e[tot].q=q;
e[tot].nxt=head[u];
head[u]=tot;
}
int f[_],dep[_],siz[_],son[_],dis[_],top[_],cnt;
void dfs1(int u,int fa) {
dep[u]=dep[fa]+1,f[u]=fa,siz[u]=1;
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(v==fa) continue;
w[v]=e[i].q;
dis[v]=dis[u]+e[i].q;
dfs1(v,u);
siz[u]+=siz[v];
if(siz[son[u]]<siz[v]) son[u]=v;
}
}
void dfs2(int u,int topf) {
top[u]=topf;
if(!son[u]) return;
dfs2(son[u],topf);
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(!top[v]) dfs2(v,v);
}
}
int lca(int x,int y) {
while(top[x]!=top[y]) {
if(dep[top[x]]<dep[top[y]]) swap(x,y);
x=f[top[x]];
} return dep[x]<dep[y]?x:y;
}
int dsr[_],js;
void dfs(int u,int fa) {
for(int i=head[u];i;i=e[i].nxt) {
int v=e[i].v;
if(v==fa) continue;
dfs(v,u);
dsr[u]+=dsr[v];
}
}
bool check(int mid) {
js=0;
for(int i=0;i<=n;++i) dsr[i]=0;
for(int i=1;i<=m;++i) {
if(len[i]<=mid) continue;
dsr[LCA[i]]-=2;
dsr[u[i]]++;
dsr[v[i]]++;
++js;
}
dfs(1,0);
int ma=0x3f3f3f3f;
for(int i=1;i<=n;++i)
if(dsr[i]==js) ma=min(ma,the_biggest-w[i]);
return ma<=mid;
}
int main() {
n=read(),m=read();
for(int i=1,x,y,q;i<n;++i) {
x=read(),y=read(),q=read();
add(x,y,q),add(y,x,q);
}
dfs1(1,0),dfs2(1,1);
for(int i=1;i<=m;++i) {
u[i]=read(),v[i]=read(),LCA[i]=lca(u[i],v[i]);
len[i]=dis[u[i]]+dis[v[i]]-2*dis[LCA[i]];
the_biggest=max(the_biggest,len[i]);
}
int l=0,r=the_biggest,ans=0;
while(l<=r) {
int mid=(l+r)>>1;
if(check(mid)) ans=mid,r=mid-1;
else l=mid+1;
} cout<<ans<<"\n";
return 0;
}

[NOIP2015]运输计划 线段树or差分二分的更多相关文章

  1. bzoj 4326: NOIP2015 运输计划【树链剖分+二分+树上差分】

    常数巨大,lg上开o2才能A 首先预处理出运输计划的长度len和lca,然后二分一个长度w,对于长度大于w的运输计划,在树上差分(d[u]+1,d[v]+1,d[lca]-2),然后dfs,找出所有覆 ...

  2. 洛谷P2680 运输计划(倍增LCA + 树上差分 + 二分答案)

    [题目链接] [思路]: 根据题意可以明显看出,当所有任务都完成时的时间是最终的结果,也就是说本题要求,求出最小的最大值. 那这样的话就暗示了将答案二分,进行check. [check方法]: 如果说 ...

  3. [NOIP2015]运输计划 D2 T3 LCA+二分答案+差分数组

    [NOIP2015]运输计划 D2 T3 Description 公元2044年,人类进入了宇宙纪元. L国有n个星球,还有n-1条双向航道,每条航道建立在两个星球之间,这n-1条航道连通了L国的所有 ...

  4. BZOJ 4326:NOIP2015 运输计划(二分+差分+lca)

    NOIP2015 运输计划Description公元 2044 年,人类进入了宇宙纪元.L 国有 n 个星球,还有 n−1 条双向航道,每条航道建立在两个星球之间,这 n−1 条航道连通了 L 国的所 ...

  5. NOIP2015 运输计划(二分+LCA+差分)

    4326: NOIP2015 运输计划 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 308  Solved: 208[Submit][Status] ...

  6. Luogu 2680 NOIP 2015 运输计划(树链剖分,LCA,树状数组,树的重心,二分,差分)

    Luogu 2680 NOIP 2015 运输计划(树链剖分,LCA,树状数组,树的重心,二分,差分) Description L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之 ...

  7. BZOJ 4326 NOIP2015 运输计划 (二分+树上差分)

    4326: NOIP2015 运输计划 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1930  Solved: 1231[Submit][Statu ...

  8. LOJ2425 NOIP2015 运输计划 【二分+LCA+树上差分】*

    LOJ2425 NOIP2015 运输计划 LINK 题意:给你一颗树,可以将任意一条边的权值变成0,然后求m条路径的长度的最小值 思路: 先二分最后的距离ans,然后我们把路程大于ans的所有路径拿 ...

  9. BZOJ 4326 NOIP2015 运输计划(树上差分+LCA+二分答案)

    4326: NOIP2015 运输计划 Time Limit: 30 Sec  Memory Limit: 128 MB Submit: 1388  Solved: 860 [Submit][Stat ...

随机推荐

  1. 关于 BenchmarkDotNet

    using BenchmarkDotNet.Attributes; using BenchmarkDotNet.Order; using System.Reflection; namespace Be ...

  2. 练手WPF(四)——贪吃蛇小游戏的简易实现(下)

    八.生成新的单节蛇身我们这里先说说游戏小原理好了,游戏运行后,通过计时器事件不断生成新的单节蛇身类SnakeNode,添加到List中的0位置,原来的蛇头变成了第二节.该节新蛇头的坐标通过蛇头前进方向 ...

  3. APS.NET MVC + EF (06)---模型

    在实际开发中,模型往往被划分为视图模型和业务模型两部分,视图模型靠近视图,业务模型靠近业务,但是在具体编码上,它们之间并不是隔离的. 6.1 视图模型和业务模型 模型大多数时候都是用来传递数据的.然而 ...

  4. Delphi - 操作Excel数据公式的实现

    procedure TF_SMP_FT_NEW.RzBitBtn_StartToChangeClick(Sender: TObject); var i, j, ni, nj, iRows, iCol, ...

  5. 页面直接导出为PDF文件,支持分页与页边距

    将WEB页面直接导出为pdf文件是经常会用到的一个功能,尤其是各种报表系统.总结了一下目前几种主流的做法: 在后端用代码生成pdf文件,比如iText一类: 在后端抓取页面并生成pdf文件,比如pha ...

  6. Python对csv排序

    #/usr/bin/evn python # -*- coding: utf-8 -*- import sys from operator import itemgetter # input_file ...

  7. 妹纸对网易严选的Bra是什么评价?

    声明:这是一篇超级严肃的技术文章,请本着学习交流的态度阅读,谢谢! 一.网易商品评论爬取 1.评论分析 进入到网易严选官网,搜索“文胸”后,先随便点进一个商品. 在商品页面,打开 Chrome 的控制 ...

  8. js中this由来

    这篇文章主要是讲述js中的this是什么?是怎么来的? 我们首先创建一个person对象,如下: var person = { name: 'wyh', age: 22, sayHi: functio ...

  9. 章节十四、9-Actions类鼠标悬停、滚动条、拖拽页面上的元素

    一.鼠标悬停 1.在web网站中,有一些页面元素只需要我们将鼠标指针放在上面就会出现被隐藏的下拉框或者其它元素,在自动化的过程中我们使用Actions类对鼠标进行悬停操作. 2.案例演示 packag ...

  10. 如何在一个ubuntu系统上搭建SVN版本控制工具

    有话说,由于公司项目部署需要,将Windows工程迁移到Linux,通过调查确定使用Ubuntu的Linux操作系统.那么如何快速搭建和Windows一样快捷方便的开发环境就很重要了.本文讲述如何在一 ...