题目大意:

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: 游戏 树链剖分+线段树的更多相关文章

  1. BZOJ2819Nim——树链剖分+线段树+Nim游戏

    题目描述 著名游戏设计师vfleaking,最近迷上了Nim.普通的Nim游戏为:两个人进行游戏,N堆石子,每回合可以取其中某一堆的任意多个,可以取完,但不可以不取.谁不能取谁输.这个游戏是有必胜策略 ...

  2. BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )

    BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...

  3. BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)

    BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...

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

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

  5. bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)

    4196: [Noi2015]软件包管理器 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 2852  Solved: 1668[Submit][Sta ...

  6. bzoj 2157: 旅游【树链剖分+线段树】

    裸的树链剖分+线段树 但是要注意一个地方--我WA了好几次才发现取完相反数之后max值和min值是要交换的-- #include<iostream> #include<cstdio& ...

  7. BZOJ 3589 动态树 (树链剖分+线段树)

    前言 众所周知,90%90\%90%的题目与解法毫无关系. 题意 有一棵有根树,两种操作.一种是子树内每一个点的权值加上一个同一个数,另一种是查询多条路径的并的点权之和. 分析 很容易看出是树链剖分+ ...

  8. 【BZOJ-2325】道馆之战 树链剖分 + 线段树

    2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1153  Solved: 421[Submit][Statu ...

  9. 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树

    [BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...

  10. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

随机推荐

  1. 多域名THINKPHP利用MEMCACHE方式共享SESSION数据(转)

    一.问题起源 稍大一些的网站,通常都会有好几个服务器,每个服务器运行着不同功能的模块,使用不同的二级域名,而一个整体性强的网站,用户系统是统一的,即一套用户名.密码在整个网站的各个模块中都是可以登录使 ...

  2. hibernate批量更新和删除数据

    批量处理  不建议用Hibernate,它的insert效率实在不搞,不过最新版本的Hibernate似乎已经在批量处理的时候做过优化了,设置一些参数如batch_size,不过性能我没有测试过,听说 ...

  3. HDU 2473 Junk-Mail Filter 删点并查集

    题目来源:pid=2473">HDU 2473 Junk-Mail Filter 题意:2中操作 M x, y 将x,y 合并到一个集合 S x 将x从所在的集合去掉 自己成为一个集合 ...

  4. XTUOJ 1176 I Love Military Chess(模拟)

     I Love Military Chess Accepted : 45   Submit : 141 Time Limit : 1000 MS   Memory Limit : 65536 KB ...

  5. 企业级API设计

    最近对service的API设计,在team内有些讨论,主要集中在API是足够抽象.通用好呢, 还是具体.易用好? 其实这个是要折衷的,通用的好处是以后更改API的可能性小,但坏处是想要通用,里面的字 ...

  6. 成为高级Java工程师,你必须要看的技术书籍

    学习的最好途径就是看书 "学习的最好途径就是看书",这是我自己学习并且小有了一定的积累之后的第一体会.个人认为看书有两点好处: 1.能出版出来的书一定是经过反复的思考.雕琢和审核的 ...

  7. 关于maven的profile

    1 什么是profile profile本质上就是不同的环境对应不同的配置. 这样的好处是,在命令行中指定具体的profile的时候,会有自己独特的参数或者独特的配置来为不同的环境生成不同的目标代码. ...

  8. php依据地理坐标获取国家、省份、城市,及周边数据类

    功能:当App获取到用户的地理坐标时,能够依据坐标知道用户当前在那个国家.省份.城市.及周边有什么数据. 原理:基于百度Geocoding API 实现.须要先注冊百度开发人员.然后申请百度AK(密钥 ...

  9. 使用了Tomcat JDBC连接池不能重连的问题

    在项目中用到了tomcat 的jdbc连接池,发现一个问题是,当数据库重启时,服务没有重新的去连接数据库,需要将部署的项目重新启动才能连接到数据库.经过测试对配置做一下修改: 在配置dataSourc ...

  10. linux复制和移动

    复制: -f  强制覆盖同名文件 -r  按递归方式保留原目录结构复制文件 cp -Rf /home/user1/*   /root/temp/ 将/home/user1目录下的所有东西拷到/root ...