bzoj 4515: 游戏 树链剖分+线段树
题目大意:
http://www.lydsy.com/JudgeOnline/problem.php?id=4515
题解:
先让我%一发lych大佬点我去看dalao的题解
讲的很详细.
这里纠正一个地方,lych大佬的式子中有一个\(a*(d[s]-d[x])+b=-a*d[x]+(b-a*d[s])\)
应该是\(a*(d[s]-d[x])+b = -a*d[x] + (b+a*d[s])\)
看来是大爷敲式子的时候敲错了...
除去这个就很完美啦
新get了线段树标记永久化的姿势
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
template<typename T>inline void read(T &x){
x=0;char ch;bool flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
const ll maxn = 210010;
const ll inf = 123456789123456789LL;
struct Edge{
ll to,next;
ll dis;
}G[maxn<<1];
ll head[maxn],cnt;
void add(ll u,ll v,ll d){
G[++cnt].to = v;
G[cnt].next = head[u];
head[u] = cnt;
G[cnt].dis = d;
}
ll son[maxn],siz[maxn],fa[maxn],top[maxn];
ll dfn[maxn],dfs_clock,num[maxn],dep[maxn];
ll d[maxn];
#define v G[i].to
void dfs(ll u){
siz[u] = 1;
for(ll i = head[u];i;i=G[i].next){
if(v == fa[u]) continue;
fa[v] = u;
d[v] = d[u] + G[i].dis;
dep[v] = dep[u] + 1;
dfs(v);
siz[u] += siz[v];
if(siz[son[u]] < siz[v]) son[u] = v;
}
}
void dfs(ll u,ll tp){
top[u] = tp;
dfn[u] = ++dfs_clock;
num[dfs_clock] = u;
if(son[u]) dfs(son[u],tp);
for(ll i = head[u];i;i=G[i].next){
if(v == fa[u] || v == son[u]) continue;
dfs(v,v);
}
}
#undef v
struct Node{
ll k,b,minn;
bool lazy;
Node(){k=lazy=0;b = minn = inf;}
}T[maxn<<2];
ll K,B,L,R;ll n;
ll query(ll rt,ll l,ll r){
if(L <= l && r <= R) return T[rt].minn;
ll ret = inf;
if(T[rt].lazy) ret = min(T[rt].k*d[num[max(L,l)]],T[rt].k*d[num[min(R,r)]]) + T[rt].b;
ll mid = l+r >> 1;
if(L <= mid) ret = min(ret,query(rt<<1,l,mid));
if(R > mid) ret = min(ret,query(rt<<1|1,mid+1,r));
return ret;
}
inline ll query(ll l,ll r){
if(l > r) swap(l,r);
L = l;R = r;
return query(1,1,n);
}
void update(ll rt,ll l,ll r,ll K,ll B){
if(T[rt].lazy == 0){
T[rt].k = K;T[rt].b = B;T[rt].lazy = 1;
if(l < r) T[rt].minn = min(T[rt<<1].minn,T[rt<<1|1].minn);
T[rt].minn = min(T[rt].minn,min(T[rt].k*d[num[l]],T[rt].k*d[num[r]])+T[rt].b);
return;
}
ll mid = l+r >> 1;
ll y1 = K*d[num[r]] + B;
ll y2 = T[rt].k*d[num[r]] + T[rt].b;
ll y3 = K*d[num[l]] + B;
ll y4 = T[rt].k*d[num[l]] + T[rt].b;
if(y3 <= y4 && y1 <= y2) T[rt].k = K,T[rt].b = B;
else if(y3 >= y4 && y1 >= y2) return;
else if(T[rt].k > K){
ll x = (B-T[rt].b)/(T[rt].k-K) + 1;
if(x <= d[num[mid]]){
swap(T[rt].b,B);swap(T[rt].k,K);
update(rt<<1,l,mid,K,B);
}else update(rt<<1|1,mid+1,r,K,B);
}else{
ll x = (T[rt].b - B - 1)/(K - T[rt].k);
if(x > d[num[mid]]){
swap(T[rt].b,B);swap(T[rt].k,K);
update(rt<<1|1,mid+1,r,K,B);
}else update(rt<<1,l,mid,K,B);
}
if(l < r) T[rt].minn = min(T[rt<<1].minn,T[rt<<1|1].minn);
if(T[rt].lazy) T[rt].minn = min(T[rt].minn,
min(T[rt].k*d[num[l]],T[rt].k*d[num[r]])+T[rt].b);
}
void ins(ll rt,ll l,ll r){
if(L <= l && r <= R){
update(rt,l,r,K,B);return;
}
ll mid = l+r >> 1;
if(L <= mid) ins(rt<<1,l,mid);
if(R > mid) ins(rt<<1|1,mid+1,r);
T[rt].minn = min(T[rt<<1].minn,T[rt<<1|1].minn);
if(T[rt].lazy) T[rt].minn = min(T[rt].minn,
min(T[rt].k*d[num[l]],T[rt].k*d[num[r]])+T[rt].b);
}
inline void ins(ll l,ll r){
if(l > r) swap(l,r);
L=l;R=r;ins(1,1,n);
}
inline ll lca(ll u,ll v){
while(top[u] != top[v]){
if(dep[top[u]] < dep[top[v]]) swap(u,v);
u = fa[top[u]];
}return dep[u] < dep[v] ? u : v;
}
inline void insert(ll u,ll v,ll a,ll b){
ll rt = lca(u,v);
K = -a;B = b+a*d[u];
while(top[u] != top[rt]){
ins(dfn[top[u]],dfn[u]);
u = fa[top[u]];
}ins(dfn[rt],dfn[u]);
K = a;B -= (d[rt]<<1)*a;
while(top[v] != top[rt]){
ins(dfn[top[v]],dfn[v]);
v = fa[top[v]];
}ins(dfn[rt],dfn[v]);
}
inline ll query_all(ll u,ll v){
ll ret = inf;
while(top[u] != top[v]){
if(dep[top[u]] < dep[top[v]]) swap(u,v);
ret = min(ret,query(dfn[top[u]],dfn[u]));
u = fa[top[u]];
}if(dep[u] < dep[v]) swap(u,v);
ret = min(ret,query(dfn[v],dfn[u]));
return ret;
}
int main(){
ll m;read(n);read(m);
for(ll i=1,u,v;i<n;++i){ll d;
read(u);read(v);read(d);
add(u,v,d);add(v,u,d);
}dfs(1);dfs(1,1);
ll op,s,t;ll a,b;
while(m--){
read(op);read(s);read(t);
if(op == 1){
read(a);read(b);
insert(s,t,a,b);
}else{
printf("%lld\n",query_all(s,t));
}
}
getchar();getchar();
return 0;
}
我能说我从昨天下午调到了今天上午,调了6个小时吗?
bzoj 4515: 游戏 树链剖分+线段树的更多相关文章
- BZOJ2819Nim——树链剖分+线段树+Nim游戏
题目描述 著名游戏设计师vfleaking,最近迷上了Nim.普通的Nim游戏为:两个人进行游戏,N堆石子,每回合可以取其中某一堆的任意多个,可以取完,但不可以不取.谁不能取谁输.这个游戏是有必胜策略 ...
- BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )
BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...
- BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)
BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...
- BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)
前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...
- bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 2852 Solved: 1668[Submit][Sta ...
- bzoj 2157: 旅游【树链剖分+线段树】
裸的树链剖分+线段树 但是要注意一个地方--我WA了好几次才发现取完相反数之后max值和min值是要交换的-- #include<iostream> #include<cstdio& ...
- BZOJ 3589 动态树 (树链剖分+线段树)
前言 众所周知,90%90\%90%的题目与解法毫无关系. 题意 有一棵有根树,两种操作.一种是子树内每一个点的权值加上一个同一个数,另一种是查询多条路径的并的点权之和. 分析 很容易看出是树链剖分+ ...
- 【BZOJ-2325】道馆之战 树链剖分 + 线段树
2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec Memory Limit: 256 MBSubmit: 1153 Solved: 421[Submit][Statu ...
- 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树
[BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...
- BZOJ2243 (树链剖分+线段树)
Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...
随机推荐
- caffe2--------ImportError: No module named past.builtins
whale@sea:~/anaconda2/lib/python2.7/site-packages$ python Python 2.7.14 |Anaconda custom (64-bit)| ( ...
- EntityFramework 6.0 修改一个已经存在的对象
public void UpdateObj(someobject obj) { db.Entry(obj).State = EntityState.Modified; db.SaveChanges() ...
- 有关SQL注入的知识
SQL注入攻击是非常令人讨厌的安全漏洞,是所有的web开发人员,不管是什么平台,技术,还是数据层,需要确信他们理解和防止的东西.不幸的是,开发人员往往不集中花点时间在这上面,以至他们的应用,更糟糕的是 ...
- iOS自己定义对象保存到本地文件
我是将聊天记录存到本地,里边用到了自己定义的对象.把数据转成Data格式存到本地.在转Data格式的时候报错了.这时候须要先将自己定义对象进行归档才干够转Data格式. 方法例如以下: 一.在.h文件 ...
- Ubuntu/CentOS下源码编译安装Php 5.6基本参数
先确认安装libxml2 apt-get install libxml2 libxml2-dev或者yum install libxml2 libxml2-dev ./configure --pref ...
- EasyPusher华为手机直播推流硬编码[OMX.IMG.TOPAZ.Encoder] failed to set input port definition parameters.
EasyPusher作为一款RTSP推送利器, 配合EasyDarwin开源流媒体服务器,在发布伊始,很快获得了广大人民群众的一致好评. 但是也有一些用户反映: EasyPusher在我的华为手机上会 ...
- Grunt是什么,以及它的一些使用方法
♥什么是Grunt Grunt和 Grunt 插件是通过 npm 安装并管理的,npm是 Node.js 的包管理器.grunt是基于node 更多插件请访问:http://www.gruntjs.n ...
- C#Panel 控件的使用
Windows 窗体 Panel 控件用于为其他控件提供可识别的分组.通常,使用面板按功能细分窗体.例如,可能有一个订单窗体,它指定邮寄选项(如使用哪一类通营承运商).将所有选项分组在一个面板中可向用 ...
- Java 中的四种引用类型(转)
目录 背景 简介 1. 强引用 StrongReference 2. 弱引用 WeakReference 3. 软引用 SoftReference ...
- HDUJ 2052 Picture 模拟
Picture Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Su ...