求树上最长路径使得经过的拥挤节点个数不超过K



//欢迎访问这个博客!http://www.cnblogs.com/luyouqi233/p/8036828.html

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
#define N 200010
using namespace std;
typedef long long ll;
typedef pair<int,int>pii;
int head[N],ecnt,dis[N],fa[N],son[N],sze[N],ans,n,m,k,que[N],t[N],nump[N],maxn[N];
//dis[i] i到根节点的距离,fa[i] 父亲,son[i] i的最大子树大小,sze[i] BFS时以i为根的子树大小(son[]和sze[])用来求重心
//t[i] 当前子树节点到子树根且经过至多i个黑点的路径的最大权值和
//nump[i] 统计i到根节点的路径上的黑点数
//maxn[i] 以根节点为起点,终点在前面已经搜过的子树中,且该路径至多经过i个黑点,这样的路径的最大权值和。
bool vis[N],iscrowd[N];//iscrowd[i] 是否是拥挤节点
inline int read()
{
int ret=0,neg=1;
char j=getchar();
for (;j>'9' || j<'0';j=getchar())
if (j=='-') neg=-1;
for (;j>='0' && j<='9';j=getchar())
ret=ret*10+j-'0';
return ret*neg;
}
struct edge
{
int nxt,v,w;
}e[2*N];
vector <pair<int,int> > s;
void add(int u,int v,int w)
{
e[++ecnt].v=v,e[ecnt].w=w,e[ecnt].nxt=head[u],head[u]=ecnt;
e[++ecnt].v=u,e[ecnt].w=w,e[ecnt].nxt=head[v],head[v]=ecnt;
}
int calcG(int sv)
{
int u,v,mx=n,G,qn;
que[qn=1]=sv,fa[sv]=0;
for (int ql=1;ql<=qn;ql++)
{
sze[u=que[ql]]=1,son[u]=0;
for (int i=head[u];i;i=e[i].nxt)
{
if (vis[v=e[i].v] || v==fa[u]) continue;
fa[v]=u,que[++qn]=v;
}
}
for (int ql=qn;ql>=1;ql--)
{
u=que[ql],v=fa[u];
if (qn-sze[u]>son[u]) son[u]=qn-sze[u];
if (son[u]<mx) G=u,mx=son[u];
if (!v) break;
sze[v]+=sze[u];
if (sze[u]>son[v]) son[v]=sze[u];
}
return G;
}
inline int getmaxp(int st,ll L)
{
int qn=0,maxp=0;
que[++qn]=st;
nump[st]=iscrowd[st];
dis[st]=L;
fa[st]=0;
maxp=max(maxp,nump[st]);
for (int ql=1,u;ql<=qn;ql++)
for (int i=head[u=que[ql]],v;i;i=e[i].nxt)
{
if (vis[v=e[i].v] || v==fa[u]) continue;
fa[v]=u;
dis[v]=dis[u]+e[i].w;
nump[v]=nump[u]+iscrowd[v];
maxp=max(maxp,nump[v]);
que[++qn]=v;
}
return maxp;
}
inline void getmaxt(int st)
{
int qn=0;
que[++qn]=st;
t[nump[st]]=max(t[nump[st]],dis[st]);
for (int ql=1,u;ql<=qn;ql++)
for (int i=head[u=que[ql]],v;i;i=e[i].nxt)
{
if (vis[v=e[i].v] || v==fa[u]) continue;
t[nump[v]]=max(t[nump[v]],dis[v]);
que[++qn]=v;
}
return ;
}
void solve(int x)
{
int G=calcG(x);
vis[G]=1;s.clear();
for (int i=head[G],v;i;i=e[i].nxt)
if (!vis[v=e[i].v])
s.push_back(pii(getmaxp(v,e[i].w),v));
sort(s.begin(),s.end());
if (iscrowd[G]) k--;
for (int i=0;i<s.size();i++)
{
getmaxt(s[i].second);
int now=0;
if (i)
{
for (int j=s[i].first;j>=0;j--)
{
while (now+j<k && now<s[i-1].first)
now++,maxn[now]=max(maxn[now],maxn[now-1]);
if(now+j<=k)ans=max(ans,maxn[now]+t[j]);
}
}
if (i!=s.size()-1)
for (int j=0;j<=s[i].first;j++)
maxn[j]=max(maxn[j],t[j]),t[j]=0;
else
for (int j=0;j<=s[i].first;j++)
{
if (j<=k)
ans=max(ans,max(t[j],maxn[j]));
maxn[j]=t[j]=0;
}
}
if (iscrowd[G]) k++;
for (int i=head[G],v;i;i=e[i].nxt)
if (!vis[v=e[i].v])
solve(v);
return ;
}
int main()
{
scanf("%d%d%d",&n,&k,&m);
for (int i=1,x;i<=m;i++)
iscrowd[read()]=1;
for (int i=1,x,y,z;i<n;i++)
x=read(),y=read(),z=read(),add(x,y,z);
solve(1);
printf("%d",ans);
return 0;
}

SPOJ 1825 Free Tour | 终极之树分治的更多相关文章

  1. SPOJ 1825 Free tour II (树的点分治)

    题目链接 Free tour II 题意:有$N$个顶点的树,节点间有权值, 节点分为黑点和白点. 找一条最长路径使得 路径上黑点数量不超过K个 这是树的点分治比较基本的题,涉及树上启发式合并……仰望 ...

  2. SPOJ:Free tour II (树分治+启发式合并)

    After the success of 2nd anniversary (take a look at problem FTOUR for more details), this 3rd year, ...

  3. SPOJ 1825 Free tour II 树分治

    题意: 给出一颗边带权的数,树上的点有黑色和白色.求一条长度最大且黑色节点不超过k个的最长路径,输出最长的长度. 分析: 说一下题目的坑点: 定义递归函数的前面要加inline,否则会RE.不知道这是 ...

  4. 【SPOJ】1825. Free tour II(点分治)

    http://www.spoj.com/problems/FTOUR2/ 先前看了一会题解就自己yy出来了...对拍过后交tle.................. 自己造了下大数据........t ...

  5. spoj 1825 Free tour II

    http://www.spoj.com/problems/FTOUR2/ After the success of 2nd anniversary (take a look at problem FT ...

  6. [spoj] FTOUR2 FREE TOUR II || 树分治

    原题 给出一颗有n个点的树,其中有M个点是拥挤的,请选出一条最多包含k个拥挤的点的路径使得经过的权值和最大. 正常树分治,每次处理路径,更新答案. 计算每棵子树的deep(本题以经过拥挤节点个数作为d ...

  7. 算法笔记--树的直径 && 树形dp && 虚树 && 树分治 && 树上差分 && 树链剖分

    树的直径: 利用了树的直径的一个性质:距某个点最远的叶子节点一定是树的某一条直径的端点. 先从任意一顶点a出发,bfs找到离它最远的一个叶子顶点b,然后再从b出发bfs找到离b最远的顶点c,那么b和c ...

  8. hdu-5977 Garden of Eden(树分治)

    题目链接: Garden of Eden Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/ ...

  9. 【BZOJ-1468】Tree 树分治

    1468: Tree Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 1025  Solved: 534[Submit][Status][Discuss] ...

随机推荐

  1. jQuery 效果使用

    .hide() 隐藏匹配的元素. .hide() 这个方法不接受任何参数. .hide([duration][,complete]) duration 一个字符串或者数字决定动画将运行多久. comp ...

  2. poj_3696_The Luckiest number

    Chinese people think of '8' as the lucky digit. Bob also likes digit '8'. Moreover, Bob has his own ...

  3. Pagehelper介绍

    本文引自:https://my.oschina.net/zudajun/blog/745232 摘要: com.github.pagehelper.PageHelper是一款好用的开源免费的Mybat ...

  4. js常用的2中排序方法:冒泡排序和快速排序

    冒泡排序:例如9 4 5 6 8 3 2 7 10 1 首先:9和4比较  4放前   4 9 5 6 8 3 2 7 10 1 4和5比较   4不动   4 9 5 6 8 3 2 7 10 1 ...

  5. 一个简单的WPF MVVM实例【转载】

    引用地址:http://blog.csdn.net/yl2isoft/article/details/20838149 1 新建WPF 应用程序WPFMVVMExample 程序结构如下图所示. 2  ...

  6. My jdbc 错误

    jdbc mysql插入数据提示Parameter index out of range (1 > number of parameters, which is 0). SqlStatement ...

  7. C语言数组篇(一)一维数组

       0.  数组的两种表现形式         一种是常见的a[10];         //初学者常用         另一种是用指针表示的数组.   //实际工程使用.常用于参数传递       ...

  8. 笔记-scrapy-深入学习-sheduler

    笔记-scrapy-深入学习-sheduler 1.      scheduler.py source code:scrapy/core/scheduler.py: 1.1.    初始化的开始 在分 ...

  9. spark&dataframe

    1.今天,我们来介绍spark以及dataframe的相关的知识点,但是在此之前先说一下对以前的hadoop的一些理解 当我启动hadoop的时候,上面有hdfs的存储结构,由于这个是分布式存储,所以 ...

  10. 直接选择排序&堆排序

    1.什么是直接选择排序? 直接选择排序(Straight Select Sort)是一种简单的排序方法,它的基本思想是:通过n-i次关键字之间的比较,从n-i+1个记录中选出关键字最小的记录,并和第i ...