【BZOJ 4515】【SDOI 2016 Round1 Day1 T3】游戏
考场上写了lct,可惜当时对标记永久化的理解并不是十分深刻,导致调一个错误的程序调了4h+,最后这道题爆0了QwQ
现在写了树链剖分,用标记永久化的线段树维护轻重链,对于$s\rightarrow lca$,$lca\rightarrow t$分开讨论,把$a×dist+b$这个式子打开,提出常数项,发现是一个一次函数(也不是严格的一次函数,只不过有单调性就可以做了)。标记永久化线段树做一下,每个节点维护一个当前区间的最小值就可以统计答案了233
这次又手残把“+”打成“*”调了一上午TwT,肉眼对拍大法好!
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long LL;
const int N = 100005;
const LL inf = 123456789123456789LL;
void read(int &k) {
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 << 1) + (k << 3) + c - '0';
k = k * fh;
} struct node {
int nxt, to, w;
} E[N << 1];
int n, m, cnt = 0, point[N], fa[N], top[N], son[N], sz[N], pos[N], wt[N], K[N << 2];
LL B[N << 2], val[N << 2], di[N], ans;
bool p[N << 2];
void ins(int x, int y, int z) {E[++cnt].nxt = point[x]; E[cnt].to = y; E[cnt].w = z; point[x] = cnt;}
void _(int x) {
sz[x] = 1;
for(int tmp = point[x]; tmp; tmp = E[tmp].nxt)
if (E[tmp].to != fa[x]) {
fa[E[tmp].to] = x;
di[E[tmp].to] = di[x] + E[tmp].w;
_(E[tmp].to);
sz[x] += sz[E[tmp].to];
if (sz[E[tmp].to] > sz[son[x]]) son[x] = E[tmp].to;
}
}
void __(int x) {
pos[x] = ++cnt; wt[cnt] = x;
if (son[x]) {top[son[x]] = top[x]; __(son[x]);}
for(int tmp = point[x]; tmp; tmp = E[tmp].nxt)
if (E[tmp].to != fa[x] && E[tmp].to != son[x])
{top[E[tmp].to] = E[tmp].to; __(E[tmp].to);}
}
int LCA(int x, int y) {
for(; top[x] != top[y]; x = fa[top[x]])
if (di[top[x]] < di[top[y]]) swap(x, y);
return di[x] < di[y] ? x : y;
}
void pushup(int rt, int l, int r) {
if (l < r) val[rt] = min(val[rt << 1], val[rt << 1 | 1]); else val[rt] = inf;
if (p[rt]) val[rt] = min(val[rt], min(di[wt[l]] * K[rt], di[wt[r]] * K[rt]) + B[rt]);
}
void Build(int rt, int l, int r) {
val[rt] = inf;
if (l == r) return;
int mid = (l + r) >> 1;
Build(rt << 1, l, mid);
Build(rt << 1 | 1, mid + 1, r);
} void update(int rt, int l, int r, int a, LL b) {
if (!p[rt]) {
p[rt] = 1; K[rt] = a; B[rt] = b;
} else {
LL newl = a * di[wt[l]] + b, newr = a * di[wt[r]] + b, befl = K[rt] * di[wt[l]] + B[rt], befr = K[rt] * di[wt[r]] + B[rt];
int mid = (l + r) >> 1;
if (newl <= befl && newr <= befr) {
K[rt] = a; B[rt] = b;
} else
if (newl >= befl && newr >= befr)
return;
else
if (a < K[rt]) {
double tmp = (b - B[rt]) / (K[rt] - a);
if (tmp <= di[wt[mid]]) {
update(rt << 1, l, mid, K[rt], B[rt]);
K[rt] = a; B[rt] = b;
} else
update(rt << 1 | 1, mid + 1, r, a, b);
} else {
double tmp = (B[rt] - b) / (a - K[rt]);
if (tmp > di[wt[mid]]) {
update(rt << 1 | 1, mid + 1, r, K[rt], B[rt]);
K[rt] = a; B[rt] = b;
} else
update(rt << 1, l, mid, a, b);
}
}
pushup(rt, l, r);
} void inc(int rt, int l, int r, int L, int R, int a, LL b) {
if (L <= l && r <= R) {update(rt, l, r, a, b); return;}
int mid = (l + r) >> 1;
if (L <= mid) inc(rt << 1, l, mid, L, R, a, b);
if (R > mid) inc(rt << 1 | 1, mid + 1, r, L, R, a, b);
pushup(rt, l, r);
}
void Q(int rt, int l, int r, int L, int R) {
if (L == l && r == R) {ans = min(ans, val[rt]); return;}
if (p[rt]) ans = min(ans, min(K[rt] * di[wt[L]], K[rt] * di[wt[R]]) + B[rt]);
int mid = (l + r) >> 1;
if (R <= mid) Q(rt << 1, l, mid, L, R);
else if (L > mid) Q(rt << 1 | 1, mid + 1, r, L, R);
else {Q(rt << 1, l, mid, L, mid); Q(rt << 1 | 1, mid + 1, r, mid + 1, R);}
} int main() {
read(n); read(m); int u, v, e, a, b, lca;
for(int i = 1; i < n; ++i) {
read(u); read(v); read(e);
ins(u, v, e); ins(v, u, e);
} _(1);
cnt = 0; top[1] = 1;
__(1);
Build(1, 1, n); LL tmp;
for(; m; --m) {
read(e);
if (e == 1) {
read(u); read(v); read(a); read(b);
lca = LCA(u, v);
tmp = 1LL * a * di[u] + b;
for(; top[u] != top[lca]; u = fa[top[u]])
inc(1, 1, n, pos[top[u]], pos[u], - a, tmp);
inc(1, 1, n, pos[lca], pos[u], - a, tmp);
tmp -= (di[lca] << 1) * a;
for(; top[v] != top[lca]; v = fa[top[v]])
inc(1, 1, n, pos[top[v]], pos[v], a, tmp);
inc(1, 1, n, pos[lca], pos[v], a, tmp);
} else {
ans = inf; read(u); read(v);
for(; top[u] != top[v]; u = fa[top[u]]) {
if (di[top[u]] < di[top[v]]) swap(u, v);
Q(1, 1, n, pos[top[u]], pos[u]);
}
if (di[u] > di[v]) swap(u, v);
Q(1, 1, n, pos[u], pos[v]);
printf("%lld\n", ans);
}
}
return 0;
}
Round2加油!
【BZOJ 4515】【SDOI 2016 Round1 Day1 T3】游戏的更多相关文章
- 【BZOJ 4598】【SDOI 2016 Round2 Day1 T3】模式字符串
2016-05-21因为BZOJ上“ 数据文件太过巨大,仅提供前三组数据测试.”所以我考场上写的60分的点分治交上去也A了. 我的这个点分治的时间复杂度是$O(Tnmlogn)$的,听题解时没听懂$O ...
- 【BZOJ 4518】【SDOI 2016 Round1 Day2 T3】征途
比较明显的斜率优化DP,省选时因为时间太紧张和斜率DP写得不熟等原因只写了60分的暴力DP,其实当时完全可以对拍来检验标算的正确,但是我当时too naive- 很快打完了,调了将近一晚上QAQ,因为 ...
- SDOI 2016 Round1 Day1
储能表 /* 引自zyz大佬的数学思想 */ #include<cstdio> #include<iostream> using namespace std; typedef ...
- 【BZOJ 4517】【SDOI 2016 Round1 Day2 T2】排列计数
本蒟蒻第一次没看题解A的题竟然是省选$Round1$ $Day2$ $T2$ 这道组合数学题. 考试时一开始以为是莫队,后来想到自己不会组合数的一些公式,便弃疗了去做第三题,,, 做完第三题后再回来看 ...
- 【NOIP 2015 & SDOI 2016 Round1 & CTSC 2016 & SDOI2016 Round2】游记
我第一次写游记,,,, 正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪里?正文在哪 ...
- [BZOJ 4516] [SDOI 2016] 生成魔咒
Description 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1.2 拼凑起来形成一个魔咒串 [1,2]. 一个魔咒串 S 的非空字串被称为魔咒串 S 的生成魔咒. 例 ...
- SDOI 2016 Round1 Day2
生成魔咒 /* 后缀数组+双向链表 参照:https://blog.csdn.net/clove_unique/article/details/53911757 */ #include<cstd ...
- 【NOIP2016】Day1 T3 换教室(期望DP)
题目背景 NOIP2016 提高组 Day1 T3 题目描述 对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程. 在可以选择的课程中,有 2n 节课程安排在 n 个时间段上. ...
- [BZOJ 1879][SDOI 2009]Bill的挑战 题解(状压DP)
[BZOJ 1879][SDOI 2009]Bill的挑战 Description Solution 1.考虑状压的方式. 方案1:如果我们把每一个字符串压起来,用一个布尔数组表示与每一个字母的匹配关 ...
随机推荐
- 【2016-10-13】【坚持学习】【Day4】【WPF】【ObservableCollection<T>】
今天在项目中使用到这个 ObservableCollection<T> 类,作为数据源集合绑定到控件. 当数据源发生变化,会通知界面显示. 如果用List<T> ,当数据源发生 ...
- java 24 - 6 GUI之 创建只能输入数字的文本框
需求: 创建一个含有标签和文本框的窗体,其中文本框只能输入数字 步骤:(大致上) 创建窗体对象 创建标签对象 创建文本框对象 把组件添加到窗体中 设置标签的监听事件,对键盘按下的数据进行监听 设置窗体 ...
- hdu 2586 How far away
How far away ? Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) T ...
- NOIP提高组2010 关押罪犯
题目描述 S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N.他们之间的关系自然也极不和谐.很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突.我们用“怨气值”(一个正整数值)来表示 ...
- javascript中的链表结构—从链表中删除元素
1.概念 上一个博文我们讲到链表,其中有一个方法remove()是暂时注释的,这个方法有点复杂,需要添加一个Previous()方法找到要删除的元素的前一个节点,这一个博文我们来分析一下这个remov ...
- iOS - 用 UIBezierPath 实现果冻效果
最近在网上看到一个很酷的下拉刷新效果(http://iostuts.io/2015/10/17/elastic-bounce-using-uibezierpath-and-pan-gesture/). ...
- Exploit利用学习1:MS09-001
目标 IP: Kali系统IP: 1.使用Metasploit框架,在kali终端输入msfconsole打开Metasploit:如下图 2.搜索相关模块:search ms09-001 找到一个可 ...
- asp.net sql 分页,,优化 排序 及分页,
调用代码: <%@ Register Assembly="AspNetPager" Namespace="Wuqi.Webdiyer" TagPrefix ...
- DEDECMS之十 修改织梦链和文章的默认来源及作者
今天在用织梦搭网站的时候,发现了两个问题,一个就是最新的dedecms5.7系统中默认会加上“织梦链”这一个链接组,织梦的做法是可以理解的, 但是给别人做网站,这些链接是不能要的,所以在数据库,模板文 ...
- 基于SignalR的小型IM系统
这个IM系统真是太轻量级了,提供的功能如下: 1.聊天内容美化 2.用户上下线提示 3.心跳包检测机制 4.加入用户可群聊 下面来一步一步的讲解具体的制作方法. 开篇准备工作 首先,巧妇难为无米之炊, ...