给定一棵 \(n\) 个点的带点权的树,求树上的路径 \(x_1,...,x_k\) ,最大化 \(\sum_{i=1}^k ia_{x_i}\)

Solution

树上路径问题可用点分治。

考虑如何合并两条路径对每条路径,记 \(l\) 为长度(点数),\(v\) 为 \(\sum_{i=1}^l ia_{x_i}\) ,\(s\) 为 \(\sum a_i\) ,那么对于两条路径 \((l_1,v_1,s_1),(l_2,v_2,s_2)\),它们的并为 \((l_1+l_2-1,v_1+s_1l_2+v_2,s_1+s_2)\)

于是答案为 \(v_1 + v_2 + s_1l_2\),其中 \(_1\) 是和之前扫描过的有关的信息,\(_2\) 是和当前正在扫描的位置有关的信息

(之所以这样取的原因是,之后李超线段树中我们需要控制下标的区间,而 \(l\) 的范围是较小的)

对于某一次查询来说,\(v_2\) 是定值,于是查询只有一个参数 \(l_2\) ,所以现在我们要最大化 \(l_2s_1 + v_1\),不妨转化为

  • 插入一条直线 \(y=ax+b\),其中 \(a=s_1,b=v_1\)
  • 询问 \(x=l_2\) 与所有直线交点中最上面的那一个

标准的李超线段树模板

我又把点分敲错,数组开小

#include <bits/stdc++.h>
using namespace std;
#define int long long const bool dbg = 0; namespace seg {
const int N = 4e6+5;
#define lc x<<1
#define rc x<<1|1
#define mid ((l+r)>>1)
struct Node {
int a,b;
} tree[N];
void ins(int x,int l,int r,int a,int b) {
if(tree[x].a*l+tree[x].b<=a*l+b&&tree[x].a*r+tree[x].b<=a*r+b) {
tree[x].a=a,tree[x].b=b;
}
else {
if(tree[x].a*l+tree[x].b>=a*l+b&&tree[x].a*r+tree[x].b>=a*r+b) return;
if(tree[x].a*mid+tree[x].b<a*mid+b)swap(tree[x].a,a),swap(tree[x].b,b);
if(a<tree[x].a)ins(lc,l,mid,a,b);
else ins(rc,mid+1,r,a,b);
}
}
void build(int x,int l,int r) {
tree[x].a=tree[x].b=0;
if(l==r)return;
build(lc,l,mid);
build(rc,mid+1,r);
}
int query(int x,int l,int r,int p) {
int ans=tree[x].a*p+tree[x].b;
if(l==r)return ans;
if(p<=mid)ans=max(ans,query(lc,l,mid,p));
else ans=max(ans,query(rc,mid+1,r,p));
return ans;
}
}; int tot;
const int N = 300005;
int n,val[N],siz[N],vis[N],mx[N],fa[N],l[N],v[N],s[N],sz,ans;
vector <int> g[N],sta; void dfs(int p) {
siz[p]=1;
mx[p]=0; //!!!
sta.push_back(p);
for(int q:g[p]) if(fa[p]!=q && !vis[q]) {
fa[q]=p;
dfs(q);
siz[p]+=siz[q];
mx[p]=max(mx[p],siz[q]);
}
} void dfs1(int p) {
if(dbg) cout<<" dfs1 "<<p<<" "<<l[p]<<" "<<v[p]<<" "<<s[p]<<endl;
seg::ins(1,1,sz,s[p],v[p]);
for(int q:g[p]) if(fa[p]!=q && !vis[q]) {
l[q]=l[p]+1;
s[q]=s[p]+val[q];
v[q]=v[p]+l[q]*val[q];
dfs1(q);
}
} void dfs2(int p) {
if(dbg) cout<<" dfs2 "<<p<<" "<<l[p]<<" "<<v[p]<<" "<<s[p]<<endl;
ans=max(ans,seg::query(1,1,sz,l[p])+v[p]);
if(dbg) cout<<" ans="<<ans<<endl;
for(int q:g[p]) if(fa[p]!=q && !vis[q]) {
l[q]=l[p]+1;
s[q]=s[p]+val[q];
v[q]=v[p]+s[q];
dfs2(q);
}
} void solve(int p) {
fa[p]=0;
sta.clear();
dfs(p);
for(int q:sta) mx[q]=max(mx[q],(int)sta.size()-siz[q]);
int r=0,mv=1e9;
for(int q:sta) if(mv>mx[q]) r=q, mv=mx[q];
vis[r]=1;
for(int q:sta) fa[q]=0;
dfs(r); //!!!
if(dbg) cout<<"solve "<<r<<endl;
sz=sta.size();
if(dbg) cout<<" round I"<<endl;
seg::build(1,1,sz);
l[r]=1; s[r]=val[r]; v[r]=val[r];
if(dbg) cout<<" "<<l[r]<<" "<<s[r]<<" "<<v[r]<<endl;
seg::ins(1,1,sz,s[r],v[r]);
for(int i=0;i<g[r].size();i++) {
int q=g[r][i];
if(vis[q]) continue;
if(dbg) cout<<" subtree "<<q<<endl;
l[q]=1;
v[q]=val[q];
s[q]=val[q];
dfs2(q);
l[q]=2;
v[q]=val[r]+val[q]*2;
s[q]=val[r]+val[q];
dfs1(q);
}
ans=max(ans,seg::query(1,1,sz,0));
if(dbg) cout<<" ans="<<ans<<endl;
seg::build(1,1,sz);
if(dbg) cout<<" round II"<<endl;
for(int i=g[r].size()-1;i>=0;--i) {
int q=g[r][i];
if(vis[q]) continue;
if(dbg) cout<<" subtree "<<q<<endl;
l[q]=1;
v[q]=val[q];
s[q]=val[q];
dfs2(q);
l[q]=2;
v[q]=val[r]+val[q]*2;
s[q]=val[r]+val[q];
dfs1(q);
}
for(int q:g[r]) if(!vis[q]) solve(q);
} signed main() {
//freopen("input.txt","r",stdin);
//freopen("output.txt","w",stdout);
//ios::sync_with_stdio(false);
cin>>n;
for(int i=1;i<n;i++) {
int t1,t2;
cin>>t1>>t2;
g[t1].push_back(t2);
g[t2].push_back(t1);
}
for(int i=1;i<=n;i++) cin>>val[i];
solve(1);
cout<<ans<<endl;
}

一杯茶一包烟,一个破题调一天

[CF1303G] Sum of Prefix Sums - 点分治,李超线段树的更多相关文章

  1. CF1303G Sum of Prefix Sums

    点分治+李超树 因为题目要求的是树上所有路径,所以用点分治维护 因为在点分治的过程中相当于将树上经过当前$root$的一条路径分成了两段 那么先考虑如何计算两个数组合并后的答案 记数组$a$,$b$, ...

  2. Codeforces 1303G - Sum of Prefix Sums(李超线段树+点分治)

    Codeforces 题面传送门 & 洛谷题面传送门 个人感觉这题称不上毒瘤. 首先看到选一条路径之类的字眼可以轻松想到点分治,也就是我们每次取原树的重心 \(r\) 并将路径分为经过重心和不 ...

  3. Codechef TSUM2 Sum on Tree 点分治、李超线段树

    传送门 点分治模板题都不会迟早要完 发现这道题需要统计所有路径的信息,考虑点分治统计路径信息. 点分治之后,因为路径是有向的,所以对于每一条路径都有向上和向下的两种.那么如果一条向上的路径,点数为\( ...

  4. Codeforces Round #463 F. Escape Through Leaf (李超线段树合并)

    听说正解是啥 set启发式合并+维护凸包+二分 根本不会啊 , 只会 李超线段树合并 啦 ... 题意 给你一颗有 \(n\) 个点的树 , 每个节点有两个权值 \(a_i, b_i\) . 从 \( ...

  5. Codeforces 1175G Yet Another Partiton Problem [DP,李超线段树]

    Codeforces 思路 首先吐槽一句:partiton是个什么东西?我好像在百度翻译里面搜不到呀qwq 发现不了什么性质,那就直接上DP吧.注意到DP可以分层,所以设\(dp_i\)表示当前层,分 ...

  6. 李超线段树(segment[HEOI2013]-洛谷T4097)

    (neng了好久好久才糊弄懂得知识点...) 一.李超线段树 在线动态维护一个二维平面直角坐标系, 支持插入一条线段, 询问与直线x = x0相交的所有线段中,交点y的最大/小值 (若有多条线段符合条 ...

  7. Codeforces 1175G - Yet Another Partiton Problem(李超线段树)

    Codeforces 题面传送门 & 洛谷题面传送门 这是一道李超线段树的毒瘤题. 首先我们可以想到一个非常 trivial 的 DP:\(dp_{i,j}\)​ 表示前 \(i\)​ 个数划 ...

  8. 【BZOJ-4515】游戏 李超线段树 + 树链剖分 + 半平面交

    4515: [Sdoi2016]游戏 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 304  Solved: 129[Submit][Status][ ...

  9. 【BZOJ-3165】Segment 李超线段树(标记永久化)

    3165: [Heoi2013]Segment Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 368  Solved: 148[Submit][Sta ...

随机推荐

  1. Idea自定义代码块【学习笔记】

    前言 idea有一个自定义代码块的功能,可以自定义代码块,方便以后工作中减少一些重复操作,这里就简单记录一下idea好用的模板吧,现在有一个关于日志的模板,用于写一个ServiceImpl方法的时候, ...

  2. Metasploit学习笔记(一) Samba服务 usermap_script安全漏洞相关信息

    一.Samba介绍 Samba是linux和unix系统上实现smb协议的一个免费软件,由客户机和服务器构成.SMB是一种在局域网上实现共享文件和打印机的协议.存在一个服务器,客户机通过该协议可以服务 ...

  3. NodeJS 介绍安装

    1.NodeJS简介 Node.js是基于Chrome JavaScript运行时建立的一个平台,实际上它是对Google Chrome V8引擎进行了封装,它主要用于创建快速的.可扩展的网络应用.N ...

  4. Nexus 安装

    Windows下安装Nexus OSS 3.12.1 1. Nexus 下载 到sonatype官网下载开源免费的OSS版本.OSS即为Open Source Software. 下载地址:https ...

  5. CentOS7下部署rsync服务

    说明: 在CentOS7下部署rsync服务和在CentOS6上部署基本上是一样的,只是CentOS7自带了rsyncd启动脚本,由systemd管理而已. rsync服务端配置 [root@SERV ...

  6. 用bootstrap来放置天气和图标的位置 自适应

    今天写了个关于天气的页面,他的摆放位置有点难,花了一两个小时用bootstrap来摆放,但是感觉bug很多 所以今天写下自己的心得,放上代码,以后这种就知道怎么写了 <div class=&qu ...

  7. AF(操作者框架)系列(2)-在Actor框架中派生Actor核心(命令模式)

    为了能够提高程序的复用性,我们准备用Actor Framework框架,来演示其满足了这个需求(本章及后面的内容,需要有OOP的基础知识). 首先,简述一下ActorFramework的运行过程: 在 ...

  8. OBS使用教程

    OBS使用教程 OBS使用教程:录屏热键的设置 视频:基础画布分辨率1920/1080 1366,768输出缩放分辨率常用FPS值 30或者60 输出:输出模式:高级录像:编码器x264码率控制 CB ...

  9. mysql 查询出现 "this is incompatible with sql_mode=only_full_group_by"错误解决方案,以及个人rpm方式重装所遇到的问题备份

    一.错误说明        这个错误发生在mysql 5.7 版本及以上版本会出现的问题:        mysql .7版本默认的sql配置是:sql_mode="ONLY_FULL_GR ...

  10. window10家庭版解决IIS中万维网服务的安全性中无Windows身份验证

    首先在左下角输入cmd搜索->命令提示符->以管理员身份运行->然后复制下面一段命令: dism /online /norestart /add-package:%SystemRoo ...