很容易想出二分这个思路,但是要想办法去掉一个 $log$.

没错,空间换时间.

双指针的部分错了好几次~

Code:

#include <set>
#include <queue>
#include <cstdio>
#include <vector>
#include <algorithm>
#define N 2000003
#define ll long long
#define setIO(s) freopen(s".in","r",stdin) , freopen(s".out","w",stdout)
using namespace std;
ll answer;
priority_queue<int>q;
vector<int>G[N],F[N],go[N];
int edges,n,root,sn,tmp,len,tt,m,flag;
int hd[N],nex[N<<1],to[N<<1],val[N<<1];
int size[N],mx[N],vis[N];
bool cmp(int a,int b)
{
return a>b;
}
void add(int u,int v,int c)
{
nex[++edges]=hd[u],hd[u]=edges,to[edges]=v,val[edges]=c;
}
void getroot(int u,int ff)
{
size[u]=1,mx[u]=0;
for(int i=hd[u];i;i=nex[i])
if(to[i]!=ff&&!vis[to[i]])
getroot(to[i],u),size[u]+=size[to[i]],mx[u]=max(mx[u],size[to[i]]);
mx[u]=max(mx[u],sn-size[u]);
if(mx[u]<mx[root]) root=u;
}
void getdis(int u,int ff,int dep)
{
F[tmp].push_back(dep);
for(int i=hd[u];i;i=nex[i])
if(to[i]!=ff&&!vis[to[i]])
getdis(to[i],u,dep+val[i]);
}
void prepare(int u)
{
vis[u]=1,G[u].push_back(++tmp),getdis(u,0,0);
for(int i=hd[u];i;i=nex[i])
if(!vis[to[i]])
{
G[u].push_back(++tmp),getdis(to[i],u,val[i]);
root=0,sn=size[to[i]],getroot(to[i],u),go[u].push_back(root),prepare(root);
}
}
ll calc(int u,int t)
{
int l=0,r=F[u].size()-1,pre=0;
if(r<=0) return 0;
ll sum=((ll)(r+1)*r)/2;
while(l<r)
{
if(F[u][l]+F[u][r]<t)
{
sum-=(ll)(r-l);
--r;
}
else
{
++l;
}
}
// printf("%lld\n",sum);
return sum;
}
void dfs(int u,int t)
{
int i;
if(!G[u].empty()) answer+=calc(G[u][0],t);
for(i=1;i<(int)G[u].size();++i) answer-=calc(G[u][i],t);
for(i=0;i<(int)go[u].size();++i) dfs(go[u][i],t);
}
ll check(int t)
{
answer=0,dfs(tt,t);
return answer;
}
multiset<int>S;
multiset<int>::iterator it;
// 长度大于 len 即可
void solve(int u)
{
int i,j;
if(!G[u].empty())
{
for(j=0;j<F[G[u][0]].size();++j)
{
if(F[G[u][0]][j]>len) q.push(F[G[u][0]][j]);
}
if(G[u].size()>=2)
{
int t=G[u][1];
for(j=0;j<F[t].size();++j) S.insert(-F[t][j]);
}
}
for(i=2;i<G[u].size();++i)
{
int t=G[u][i];
// F[t] 与 S 结合
it=S.begin();
for(j=0;j<F[t].size()&&(-(*S.begin())+F[t][0])>len;++j)
{
for(it=S.begin();it!=S.end();it++)
{
if(-(*it)+F[t][j]>len)
{
q.push(-(*it)+F[t][j]);
}
else break;
}
}
for(j=0;j<F[t].size();++j)
S.insert(-F[t][j]);
}
S.clear();
for(int i=0;i<go[u].size();++i) solve(go[u][i]);
}
int main()
{
int i,j,tot=0;
// setIO("input");
scanf("%d%d",&n,&m);
for(i=1;i<n;++i)
{
int a,b,c;
scanf("%d%d%d",&a,&b,&c),add(a,b,c),add(b,a,c),tot+=c;
}
mx[root=0]=sn=n,getroot(1,0),tt=root,prepare(root);
for(i=1;i<=tmp;++i) sort(F[i].begin(),F[i].end(),cmp);
int l=0,r=tot,mid,ans=0; while(l<=r)
{
mid=(l+r)>>1;
if(check(mid)>=1ll*m) ans=mid,l=mid+1;
else r=mid-1;
}
len=ans,solve(tt);
while(q.size()<m) q.push(len);
while(!q.empty()) printf("%d\n",q.top()), q.pop();
return 0;
}

  

BZOJ 3784: 树上的路径 点分治+二分+set的更多相关文章

  1. BZOJ.3784.树上的路径(点分治 贪心 堆)

    BZOJ \(Description\) 给定一棵\(n\)个点的带权树,求树上\(\frac{n\times(n-1)}{2}\)条路径中,长度最大的\(m\)条路径的长度. \(n\leq5000 ...

  2. bzoj 3784: 树上的路径 堆维护第k大

    3784: 树上的路径 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 88  Solved: 27[Submit][Status][Discuss] ...

  3. bzoj 3784: 树上的路径【点分治+st表+堆】

    参考:https://www.cnblogs.com/CQzhangyu/p/7071477.html 神奇的点分治序(或者叫点剖?).就是把点分治扫过的点依次放进队列里,然后发现,对于每一棵树摊到序 ...

  4. BZOJ 3784: 树上的路径

    Description 问一棵树上前 \(k\) 大路径的边权. Sol 边分治. 非常感谢数据没有菊花图. 为了写写边分治试试然后就开了这道题. 边分治非常好想,选一条重边,分成两部分,然后分别求最 ...

  5. 【BZOJ-3784】树上的路径 点分治 + ST + 堆

    3784: 树上的路径 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 462  Solved: 153[Submit][Status][Discuss ...

  6. 【BZOJ3784】树上的路径 点分治序+ST表

    [BZOJ3784]树上的路径 Description 给定一个N个结点的树,结点用正整数1..N编号.每条边有一个正整数权值.用d(a,b)表示从结点a到结点b路边上经过边的权值.其中要求a< ...

  7. BZOJ 1316: 树上的询问( 点分治 + 平衡树 )

    直接点分治, 用平衡树(set就行了...)维护. -------------------------------------------------------------------------- ...

  8. BZOJ 3697: 采药人的路径 [点分治] [我想上化学课]

    传送门 题意: 路径有$-1,1$两种权值,求有多少路径满足权值和为$0$且有一个点将路径分成权值和为$0$的两段 第四节课本来想去上化学,然后快上课了这道题还没调出来.....可恶我想上化学 昨天两 ...

  9. BZOJ 3697: 采药人的路径 点分治

    好久不做点分治的题了,正好在联赛之前抓紧复习一下. 先把边权为 $0$ 的置为 $-1$.定义几个状态:$f[dis][0/1],g[dis][0/1]$ 其中 $f$ 代表在当前遍历的子树内的答案. ...

随机推荐

  1. python r r+;w w+;a a+;以及加不加b区别

    #以下内容均在正常打开文件的情况下运行 一.列表格 模式 可做操作 若文件不存在 是否覆盖 r 只能读 报错 --- r+ 可读可写 报错 是 w 只能写 创建 是 w+ 可读可写 创建 是 a 只能 ...

  2. Serialize and Deserialize N-ary Tree

    Serialization is the process of converting a data structure or object into a sequence of bits so tha ...

  3. 【转帖】网卡多队列技术与RSS功能介绍

    网卡多队列技术与RSS功能介绍 2017年02月08日 15:44:37 Murphy_0806 阅读数 10665 标签: rss网卡dpdk 更多 个人分类: DPDK https://blog. ...

  4. NoSQL数据库简介与产生

    关系型数据库所存在“问题” >利用ACID原则(原子性,一致性,隔离性,持久性)保证数据完整性: >行列的规范化存储: >预定义结构: >存储数据量“小”: >结构化查询 ...

  5. JavaScript处理股票数据

    1, 先使用Ajax发送异步请求到:http://hq.sinajs.cn/list=s_sh000001 2, 然后用[,]切割成数组https://www.w3school.com.cn/js/j ...

  6. 一头扎进 JAVA

    硅不可 吉米 JAVA 基础 -- 基础不牢,地动山摇 子类应该比 父类更为 开放 (public protected default private) 子类方法不能比父类抛出更高异常( 可以为父类方 ...

  7. OpenSSL(2)创建私有证书颁发机构

    如果想要建立自己的CA, OpenSSL已经包含了所有你需要的东西.所有的操作都通过纯命令行执行,虽然不那么友好,整个过程也比较长,但是这可以让你去思考每个细节. 我建议自己创建一套私有的 CA主要是 ...

  8. mongodb的简单操作记录

    由于项目上需要对mongodb进行监控,所以需要先熟悉下什么是mongobd以及mongodb的简单操作 mongodb的安装: curl -O https://fastdl.mongodb.org/ ...

  9. HashMap原理探究

    一.写随笔的原因:HashMap我们在平时都会用,一般面试题也都会问,借此篇文章分析下HashMap(基于JDK1.8)的源码. 二.具体的内容: 1.简介: HashMap在基于数组+链表来实现的, ...

  10. 3.1.2-arm-linux-ld选项

    有文件link.S,内容如下 .text .global _start _start: b step1 step1: ldr pc, =step2 step2: b step2 经过如下命令编译 ar ...