博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Codeforces Round #463 F. Escape Through Leaf (李超线段树合并)
阅读量:4616 次
发布时间:2019-06-09

本文共 3841 字,大约阅读时间需要 12 分钟。

听说正解是啥 set启发式合并+维护凸包+二分 根本不会啊 , 只会 李超线段树合并 啦 ...

题意

给你一颗有 \(n\) 个点的树 , 每个节点有两个权值 \(a_i, b_i\) .

\(u\) 跳到 \(v\) 的代价是 \(a_u \times b_v\) . 你需要计算每个节点跳到叶子的最小代价 .

\((n \le 10^5, -10^5 \le a_i, b_i \le 10^5)\)

题解

我们首先考虑一个很容易的 \(dp\) , 令 \(dp_i\)\(i\) 跳到叶子的最小代价 .

那么显然有一个转移 此处 \(v\)\(u\) 的后代 .

\[\displaystyle dp_u = \min_v \{a[u] \times b[v] + dp_v\}\]

暴力转移是 \(O(n^2)\) 的显然无法接受 .

那么考虑优化 , 不难发现这个转移就是 李超线段树上求多条直线 \(y=kx+b\)\(x=k\) 最值的形式 .

\((k = b[v], x = a[u], b=dp_v)\)

那么显然可以考虑用李超线段树维护这个 \(dp\) .

对于树上的每个点 , 可以用一颗李超线段树维护这个点子树的所有直线信息 .

然后我们只需要考虑合并几颗子树信息了 , 不难发现是 套路的 线段树合并 . (这样时间空间复杂度都正确了?)

我们直接同时遍历两颗线段树 , 然后把其中一颗当前区间的优势直线暴力插入另外一颗线段树 .

最后把子树全都合并上来后 , 直接询问出 \(dp_u\) 就行了 , 然后再插入到这颗线段树中去 .

由于询问 \(a_i\) 可能为负数 , 而线段树不太好维护负数 , 我们考虑插入直线和询问的时候都向右平移 \(lim = 10^5\) 长度 .

原来的直线 \(y=kx+b\) 就变成了 \(y'=k(x-lim)+b=kx+(b-k\cdot lim)\) 了 .

(也许可以维护负数 diversion 说可以)

最后瞎分析时间复杂度 \(O(\sum minsize \log n) = O(n \log ^2 n)\) ... (错了的话大佬帮我指正啊qwq)

这道题还是比较好写的 ...

代码

#include 
#define For(i, l, r) for(register int i = (l), i##end = (int)(r); i <= i##end; ++i)#define Fordown(i, r, l) for(register int i = (r), i##end = (int)(l); i >= i##end; --i)#define Set(a, v) memset(a, v, sizeof(a))#define debug(x) cout << #x << ':' << x << endlusing namespace std;inline bool chkmin(int &a, int b) {return b < a ? a = b, 1 : 0;}inline bool chkmax(int &a, int b) {return b > a ? a = b, 1 : 0;}inline int read() { int x = 0, fh = 1; char ch = getchar(); for (; !isdigit(ch); ch = getchar()) if (ch == '-') fh = -1; for (; isdigit(ch); ch = getchar()) x = (x << 1) + (x << 3) + (ch ^ 48); return x * fh;}void File() {#ifdef zjp_shadow freopen ("F.in", "r", stdin); freopen ("F.out", "w", stdout);#endif}const int N = 2e5 + 1e3, Lim = 1e5 + 5;typedef long long ll;struct Line { ll k, b; int id; ll func(int x) { return k * x + b; } };inline bool Cmp(Line a, Line b, int x) { if (!a.id) return true; return a.func(x) > b.func(x); }int rt[N];const int Maxn = N * 30;#define lson ls[o], l, mid#define rson rs[o], mid + 1, rstruct Chao_Segment_Tree { Line Adv[Maxn]; int ls[Maxn], rs[Maxn], Size; Chao_Segment_Tree () {Size = 0;}; void Insert(int &o, int l, int r, Line uv) { if (!o) o = ++ Size; int mid = (l + r) >> 1; if (Cmp(Adv[o], uv, mid)) swap(Adv[o], uv); if (l == r || Adv[o].k == uv.k || !uv.id) return ; double x = (double)(Adv[o].b - uv.b) / (uv.k - Adv[o].k); if (x < l || x > r) return ; if (uv.k > Adv[o].k) Insert(lson, uv); else Insert(rson, uv); } int Merge(int x, int y, int l, int r) { if (!x || !y) return x + y; Insert(x, l, r, Adv[y]); int mid = (l + r) >> 1; ls[x] = Merge(ls[x], ls[y], l, mid); rs[x] = Merge(rs[x], rs[y], mid + 1, r); return x; } Line Query(int o, int l, int r, int qp) { if (l == r) return Adv[o]; int mid = (l + r) >> 1; Line tmp = (qp <= mid) ? Query(lson, qp) : Query(rson, qp); return Cmp(tmp, Adv[o], qp) ? Adv[o] : tmp; }} T;int n, A[N], B[N]; vector
G[N];ll dp[N];inline void Insert(int ver) { ll k = B[ver], b = dp[ver] - Lim * k; T.Insert(rt[ver], 1, Lim * 2, (Line) {k, b, ver});}inline ll Query(int ver) { int tmp = T.Query(rt[ver], 1, Lim * 2, A[ver] + Lim).id; return 1ll * A[ver] * B[tmp] + dp[tmp];}void Dp(int u, int fa) { for (int v : G[u]) if (v ^ fa) Dp(v, u), rt[u] = T.Merge(rt[u], rt[v], 1, Lim * 2); dp[u] = Query(u); Insert(u);}int main () { File(); n = read(); For (i, 1, n) A[i] = read(); For (i, 1, n) B[i] = read(); For (i, 1, n - 1) { int u = read(), v = read(); G[u].push_back(v); G[v].push_back(u); } Dp(1, 0); For (i, 1, n) printf ("%lld%c", dp[i], i == iend ? '\n' : ' '); return 0;}

转载于:https://www.cnblogs.com/zjp-shadow/p/9180474.html

你可能感兴趣的文章
HTTP请求过程
查看>>
织梦多域名解析到同一个空间导致打开链接不一致怎么办?
查看>>
OpenCV探索之路(十五):角点检测
查看>>
Xcode10 library not found for -lstdc++ 找不到问题
查看>>
Mysql 8.0.13如何重置密码
查看>>
关于Ubuntu10.04磁盘空间不足的问题
查看>>
Windows Phone 7 Belling‘s课堂(九) LINQ to SQL语句
查看>>
python协程函数、递归、匿名函数与内置函数使用、模块与包
查看>>
[ConcurrencyCheck]并发检查
查看>>
发布功能完成
查看>>
excel 合并单元格
查看>>
ASP.NET Core 使用 URL Rewrite 中间件实现 HTTP 重定向到 HTTPS
查看>>
Ubuntu Server无法安装busybox-initramfs
查看>>
List
查看>>
iOS设计模式简介
查看>>
HTML
查看>>
c# 扩展方法 奇思妙用 高级篇 九:OrderBy(string propertyName, bool desc)
查看>>
Log4net入门(ASP.NET MVC 5篇)
查看>>
C语言中的地址传递(传指针,传递给形参的指针仍然是实参指针的一份拷贝)
查看>>
redis缓存数据库及Python操作redis
查看>>