http://www.lydsy.com/JudgeOnline/problem.php?id=3672

http://uoj.ac/problem/7

链上的情况可以用斜率优化dp。树上用斜率优化dp时,单调队列的复杂度是均摊$O(n)$的,所以放到树上做“可持久化单调队列”复杂度是$O(n^2)$的,所以不能树上斜率优化。

这道题可以用树链剖分(时间复杂度$O(n\log^3n)$)或者点分治套cdq分治(时间复杂度$O(n\log^2n)$)。因为树链剖分感觉比较难写,而且每个节点用vector存单调队列,显得比较卡空间,而且时间复杂度多一个log,所以写了点分治。

对于一个点$i$,从i到根的路径上有$j$,$k$。假设$k$的深度比$j$小,且用$k$来更新$i$比$j$更优,得出式子($dis$为到根的距离):

$$\frac{f_j-f_k}{dis_j-dis_k}>p_i$$

对于每个点,把$dis$看成横坐标,$f$看成纵坐标,最优点一定在下凸壳上。

对于一棵树,求出分治重心,再对重心上方的子树进行分治,分治完后重心到该树的根上所有的点的$f$值都求好了,然后就用重心到根这条链上所有的点去更新重心子树中所有的点。

先对重心子树中所有的点(不包括重心)按“$l$值-到重心的距离”排序,保证用来更新“子树中的点”的“重心到根上的点”到重心的距离单调递增,这样拿单调栈来维护一个下凸壳来更新就可以了。

时间复杂度$O(n\log^2n)$。

#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
const int N = 200003;
int in() {
int k = 0, fh = 1; char c = getchar();
for(; c < '0' || c > '9'; c = getchar())
if (c == '-') fh = -1;
for(; c >= '0' && c <= '9'; c = getchar())
k = (k << 3) + (k << 1) + c - '0';
return k * fh;
}
ll inll() {
ll k = 0; int fh = 1; char c = getchar();
for(; c < '0' || c > '9'; c = getchar())
if (c == '-') fh = -1;
for(; c >= '0' && c <= '9'; c = getchar())
k = (k << 3) + (k << 1) + c - '0';
return k * fh;
} bool vis[N];
struct node {
int nxt, to; ll w;
node(int _nxt = 0, int _to = 0, ll _w = 0) : nxt(_nxt), to(_to), w(_w) {}
} E[N << 1];
int t, n, fa[N], Q[N], cnt = 0, sz[N], point[N];
ll f[N], fadis[N], p[N], q[N], l[N], longdis[N]; void ins(int u, int v, ll w) {E[++cnt] = node(point[u], v, w); point[u] = cnt;} int findrt(int x) {
int u, head = 0, tail = 1; Q[1] = x;
while (head != tail) {
u = Q[++head]; sz[u] = 1;
for(int i = point[u]; i; i = E[i].nxt)
if (!vis[E[i].to] && E[i].to != fa[u])
Q[++tail] = E[i].to;
}
for(int i = tail; i >= 1; --i) {
if ((sz[Q[i]] << 1) > tail) return Q[i];
sz[fa[Q[i]]] += sz[Q[i]];
}
} int tot, qu[N];
struct data {
ll dis, line; int id;
data(ll _dis = 0, ll _line = 0, int _id = 0) : dis(_dis), line(_line), id(_id) {}
bool operator < (const data &A) const {
return line < A.line;
}
} a[N]; double k_num(int x, int y) {
return 1.0 * (f[x] - f[y]) / (longdis[x] - longdis[y]);
} int find(int le, double k) {
int left = 0, right = le - 1, mid;
while (left < right) {
mid = (left + right) >> 1;
if (k_num(qu[mid], qu[mid + 1]) > k) left = mid + 1;
else right = mid;
}
if (left == le - 1 && k_num(qu[left], qu[le]) > k) return qu[le];
return qu[left];
} int cont = 0;
void cdq(int x) {
vis[x] = true;
int i, tmp, tail, head = 0, up = x; ll len = 0;
for(tmp = point[x]; tmp; tmp = E[tmp].nxt)
if (!vis[E[tmp].to] && E[tmp].to == fa[x]) {
while (!vis[fa[up]] && up != 1) up = fa[up];
cdq(findrt(up));
break;
} tot = 0;
for(i = point[x]; i; i = E[i].nxt)
if (!vis[E[i].to] && E[i].to != fa[x])
a[++tot] = data(fadis[E[i].to], l[E[i].to] - fadis[E[i].to], E[i].to); while (head != tot) {
++head;
for(i = point[a[head].id]; i; i = E[i].nxt)
if (!vis[E[i].to] && E[i].to != fa[a[head].id])
a[++tot] = data(fadis[E[i].to] + a[head].dis, l[E[i].to] - fadis[E[i].to] - a[head].dis, E[i].to);
} stable_sort(a + 1, a + tot + 1); tmp = x;
while (tmp != up) {
tmp = fa[tmp]; if (longdis[x] - longdis[tmp] > l[x]) break;
if (f[x] == -1) f[x] = f[tmp] + (longdis[x] - longdis[tmp]) * p[x] + q[x];
else f[x] = min(f[x], f[tmp] + (longdis[x] - longdis[tmp]) * p[x] + q[x]);
} tail = 0; tmp = x; qu[0] = x;
for(i = 1; i <= tot; ++i) {
if (a[i].line < 0) continue;
while (tmp != up && len + fadis[tmp] <= a[i].line) {
len += fadis[tmp];
tmp = fa[tmp];
while (tail && k_num(tmp, qu[tail]) > k_num(qu[tail], qu[tail - 1]))
--tail;
qu[++tail] = tmp;
}
head = find(tail, (double) p[a[i].id]);
if (f[a[i].id] == -1) f[a[i].id] = f[head] + (longdis[a[i].id] - longdis[head]) * p[a[i].id] + q[a[i].id];
else f[a[i].id] = min(f[a[i].id], f[head] + (longdis[a[i].id] - longdis[head]) * p[a[i].id] + q[a[i].id]);
} for(i = point[x]; i; i = E[i].nxt)
if (!vis[E[i].to] && E[i].to != fa[x])
cdq(findrt(E[i].to));
} int main() {
n = in(); t = in();
for(int i = 2; i <= n; ++i) {
fa[i] = in(); fadis[i] = inll();
ins(fa[i], i, fadis[i]);
ins(i, fa[i], fadis[i]);
p[i] = inll(); q[i] = inll(); l[i] = inll();
} memset(f, -1, sizeof(ll) * (n + 1));
f[1] = 0;
int u, head = 0, tail = 1;
Q[1] = 1; longdis[1] = 0;
while (head != tail) {
u = Q[++head];
for(int i = point[u]; i; i = E[i].nxt)
if (E[i].to != fa[u]) {
longdis[E[i].to] = longdis[u] + fadis[E[i].to];
Q[++tail] = E[i].to;
}
} cdq(findrt(1));
for(int i = 2; i <= n; ++i)
printf("%lld\n", f[i]);
return 0;
}

终于调完了,写完后有好多错QAQ

【BZOJ 3672】【UOJ #7】【NOI 2014】购票的更多相关文章

  1. [LOJ 2720][BZOJ 5417][UOJ 395][NOI 2018]你的名字

    [LOJ 2720][BZOJ 5417][UOJ 395][NOI 2018]你的名字 题意 给定一个大串 \(S\) 以及 \(q\) 次询问, 每次询问给定一个串 \(T\) 和区间 \([l, ...

  2. BZOJ 3672 NOI 2014 购票

    题面 Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. 全国的城市构成了一棵以SZ市为根的有根树,每个城市 ...

  3. 【BZOJ 3669】【NOI 2014】魔法森林 LCT+枚举边

    $LCT+枚举$ 复习一下$LCT$模板. 先以$Ai$为关键字$sort$,然后$Ai$从小到大枚举每条边,看能否构成环,构不成则加边,构成则判断,判断过了就切断$Bi$最大的边. 我的边是编号为$ ...

  4. 解题:NOI 2014 购票

    题面 观察一下部分分,我们发现链上的部分分是这样一个DP: $dp[i]=min(dp[i],dp[j]+dis(i,j)*p[i]+q[i])(dis(i,j)<=lim[i]\&\& ...

  5. BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)

    前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...

  6. [luogu P2375] [NOI 2014] 动物园

    [luogu P2375] [NOI 2014] 动物园 题目描述 近日,园长发现动物园中好吃懒做的动物越来越多了.例如企鹅,只会卖萌向游客要吃的.为了整治动物园的不良风气,让动物们凭自己的真才实学向 ...

  7. 【BZOj 3670】【UOJ #5】【NOI 2014】动物园

    http://www.lydsy.com/JudgeOnline/problem.php?id=3670 http://uoj.ac/problem/5 可以建出"KMP自动机"然 ...

  8. bzoj 3672: [Noi2014]购票 树链剖分+维护凸包

    3672: [Noi2014]购票 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 480  Solved: 212[Submit][Status][D ...

  9. BZOJ 3672 购票

    Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国\(n\)个城市的OIer们都会从各地出发,到SZ市参加这次盛会. 全国的城市构成了一棵以SZ市为根的有根树,每个城市与 ...

随机推荐

  1. stm32中断无电平触发的解决办法

    这几天在用stm32读取FPGA中FIFO里的数据,遇到了不少的问题.其中有个自己觉得比较好玩的问题,就拿出来写写.其实这个问题也比较简单,开始我觉得没必要拿出来写,不过,想想后觉得还是写写吧,就当做 ...

  2. POJ3783Balls[DP 最坏情况最优解]

    Balls Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 907   Accepted: 598 Description T ...

  3. WCF添加服务失败一则

    原因是本机开发IIS没有安装HTTPS证书 将红色的字注释掉就好了! <services> <service behaviorConfiguration="basicSer ...

  4. ANE原生代码的调试(安卓)

    忙了一天终于有空继续这篇教程了. ANE的原生代码的调试其实在Adobe的官网有介绍的,但是同样很含糊,我摸索了一段时间现在记录下我的心得. 首先你得安装Eclipse,然后你得启动Eclipse 然 ...

  5. Java中的链表数据结构

    首先,我们来定义一个链表的数据结构,如下: 1 public class Link { 2 private int value; 3 private Link next; 4 public void ...

  6. Java8简单的本地缓存实现

    原文出处:lukaseder         Java8简单的本地缓存实现 这里我将会给大家演示用ConcurrentHashMap类和lambda表达式实现一个本地缓存.因为Map有一个新的方法,在 ...

  7. 重构Web Api程序(Api Controller和Entity) 续篇(2)

    一代好的程序,是几经修改与重构出来的.有关此篇Insus.NET对前面所在修改与重构,还是没有觉得很完美简洁.特别对是存在的json文件进行修改或是删除时,我们原先是从文件读取数据,修改或是删除数据, ...

  8. UML:类图复习-鸡生蛋,蛋生鸡

    这是前一阵<高级软件工程>课堂上,老师随堂出的一道讨论题,随手贴在这里: ps: 今天是520,正好聊一些OoXx,关于爱的扯淡话题:) 题目:“鸡生蛋,蛋孵鸡”,世间万物生生不息,如何用 ...

  9. hdu5444Elven Postman(主席树思想的应用)

    主席树这个概念应该不陌生吧!恩?不会, 戳这里. 主席树(函数式线段树)用的是函数思想,一个节点开数组用来保存自己的左右节点,这样节省许多不必要的空间,还可以保存许多历史状态.而这里我们用的是主席树的 ...

  10. Java实现生产者和消费者

    生产者和消费者问题是操作系统的经典问题,在实际工作中也常会用到,主要的难点在于协调生产者和消费者,因为生产者的个数和消费者的个数不确定,而生产者的生成速度与消费者的消费速度也不一样,同时还要实现生产者 ...