做了 [JSOI2008]Blue Mary开公司 以后发现这 tm 不就是个傻逼树剖+李超线段树吗,做了以后发现我才是傻逼……树剖竟然写错了……这题是我目前写过最长的代码了qwq

#include <iostream>
#include <cstdio>
using namespace std;
typedef long long ll;
int n, m, uu, vv, ww, dep[100005], fa[100005], dfn[100005], son[100005], siz[100005];
int top[100005], idx, dui[100005], opt, ss, tt, cnt, hea[100005];
ll dis[100005];
const ll oo=123456789123456789;
struct Edge{
int too, nxt, val;
}edge[200005];
struct SGT{
ll val[400005], vak[400005], bva[400005];
bool vis[400005];
void pushUp(int o, int lson, int rson){
val[o] = min(val[o], min(val[lson], val[rson]));
}
void build(int o, int l, int r){
val[o] = oo;
if(l==r) ;
else{
int mid=(l+r)>>1;
int lson=o<<1;
int rson=lson|1;
if(l<=mid) build(lson, l, mid);
if(mid<r) build(rson, mid+1, r);
}
}
void updateLichao(int o, int l, int r, ll k, ll b){
ll lvalnow=k*dis[dui[l]]+b;
ll rvalnow=k*dis[dui[r]]+b;
ll lvalpre=vak[o]*dis[dui[l]]+bva[o];
ll rvalpre=vak[o]*dis[dui[r]]+bva[o];
if(!vis[o]){
vis[o] = true;
vak[o] = k; bva[o] = b;
val[o] = min(val[o], min(lvalnow, rvalnow));
return ;
}
else if(lvalnow>=lvalpre && rvalnow>=rvalpre) return ;
else if(lvalnow<lvalpre && rvalnow<rvalpre){
val[o] = min(val[o], min(lvalnow, rvalnow));
vak[o] = k;
bva[o] = b;
}
else{
int mid=(l+r)>>1;
int lson=o<<1;
int rson=lson|1;
ll mvalnow=k*dis[dui[mid]]+b;
ll mvalpre=vak[o]*dis[dui[mid]]+bva[o];
if(lvalnow>=lvalpre){
if(mvalnow>=mvalpre) updateLichao(rson, mid+1, r, k, b);
else{
updateLichao(lson, l, mid, vak[o], bva[o]);
vak[o] = k;
bva[o] = b;
val[o] = min(val[o], min(lvalnow, rvalnow));
}
}
else{
if(mvalnow>=mvalpre) updateLichao(lson, l, mid, k, b);
else{
updateLichao(rson, mid+1, r, vak[o], bva[o]);
vak[o] = k;
bva[o] = b;
val[o] = min(val[o], min(lvalnow, rvalnow));
}
}
pushUp(o, lson, rson);
}
}
void update(int o, int l, int r, int x, int y, ll k, ll b){
if(l>=x && r<=y)
updateLichao(o, l, r, k, b);
else{
int mid=(l+r)>>1;
int lson=o<<1;
int rson=lson|1;
if(x<=mid) update(lson, l, mid, x, y, k, b);
if(mid<y) update(rson, mid+1, r, x, y, k, b);
pushUp(o, lson, rson);
}
}
ll query(int o, int l, int r, int x, int y){
if(l>=x && r<=y) return val[o];
else{
int mid=(l+r)>>1;
int lson=o<<1;
int rson=lson|1;
ll re=oo;
if(vis[o]) re = min(re, min(vak[o]*dis[dui[max(l,x)]]+bva[o], vak[o]*dis[dui[min(r,y)]]+bva[o]));
if(x<=mid) re = min(re, query(lson, l, mid, x, y));
if(mid<y) re = min(re, query(rson, mid+1, r, x, y));
return re;
}
}
}sgt;
void add_edge(int fro, int too, int val){
edge[++cnt].nxt = hea[fro];
edge[cnt].too = too;
edge[cnt].val = val;
hea[fro] = cnt;
}
void dfs1(int x, int f, ll p){
dep[x] = dep[f] + 1;
dis[x] = p;
fa[x] = f;
siz[x] = 1;
int maxSon=-1;
for(int i=hea[x]; i; i=edge[i].nxt){
int t=edge[i].too;
if(t!=f){
dfs1(t, x, p+edge[i].val);
siz[x] += siz[t];
if(siz[t]>maxSon){
son[x] = t;
maxSon = siz[t];
}
}
}
}
void dfs2(int x, int topf){
dfn[x] = ++idx;
top[x] = topf;
dui[idx] = x;
if(!son[x]) return ;
dfs2(son[x], topf);
for(int i=hea[x]; i; i=edge[i].nxt){
int t=edge[i].too;
if(t!=fa[x] && t!=son[x])
dfs2(t, t);
}
}
int getLca(int u, int v){
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]) swap(u, v);
u = fa[top[u]];
}
if(dep[u]>dep[v]) swap(u, v);
return u;
}
void rangeUpdate(){
int lca=getLca(ss, tt);
int x=ss;
ll nb=uu*dis[x]+vv;
ll nk=-uu;
while(top[x]!=top[lca]){
sgt.update(1, 1, n, dfn[top[x]], dfn[x], nk, nb);
x = fa[top[x]];
}
sgt.update(1, 1, n, dfn[lca], dfn[x], nk, nb);
nb = uu*(dis[ss]-dis[lca]-dis[lca])+vv;
nk = uu;
x = tt;
while(top[x]!=top[lca]){
sgt.update(1, 1, n, dfn[top[x]], dfn[x], nk, nb);
x = fa[top[x]];
}
sgt.update(1, 1, n, dfn[lca], dfn[x], nk, nb);
}
ll rangeQuery(){
ll re=oo;
while(top[ss]!=top[tt]){
if(dep[top[ss]]<dep[top[tt]]) swap(ss, tt);
re = min(re, sgt.query(1, 1, n, dfn[top[ss]], dfn[ss]));
ss = fa[top[ss]];
}
if(dep[ss]>dep[tt]) swap(ss, tt);
re = min(re, sgt.query(1, 1, n, dfn[ss], dfn[tt]));
return re;
}
int main(){
cin>>n>>m;
for(int i=1; i<n; i++){
scanf("%d %d %d", &uu, &vv, &ww);
add_edge(uu, vv, ww);
add_edge(vv, uu, ww);
}
dfs1(1, 0, 0);
dfs2(1, 1);
sgt.build(1, 1, n);
while(m--){
scanf("%d", &opt);
if(opt==1){
scanf("%d %d %d %d", &ss, &tt, &uu, &vv);
rangeUpdate();
}
else{
scanf("%d %d", &ss, &tt);
printf("%lld\n", rangeQuery());
}
}
return 0;
}

loj2032 「SDOI2016」游戏的更多相关文章

  1. 【LOJ】#2032. 「SDOI2016」游戏

    题解 看错题了,以为单次修改相当于一个覆盖,后来才明白"添加"-- 就相当于添加很多线段求最小值 首先这个等差数列添加的方式比较烦人,我们拆开两条链,一条s到lca,一条lca到t ...

  2. LOJ_2305_「NOI2017」游戏 _2-sat

    LOJ_2305_「NOI2017」游戏 _2-sat 题意: 给你一个长度为n的字符串S,其中第i个字符为a表示第i个地图只能用B,C两种赛车,为b表示第i个地图只能用A,C两种赛车,为c表示第i个 ...

  3. 「SDOI2016」储能表(数位dp)

    「SDOI2016」储能表(数位dp) 神仙数位 \(dp\) 系列 可能我做题做得少 \(QAQ\) \(f[i][0/1][0/1][0/1]\) 表示第 \(i\) 位 \(n\) 是否到达上界 ...

  4. 「HNOI2018」游戏

    「HNOI2018」游戏 解题思路 首先没有锁上的门可以缩点缩掉,然后对于一扇锁上的门,如果钥匙在左边,那么右边就永远不可能到达左边,同理如果钥匙在右边,左边就永远不可能到达右边. 然后考虑一个暴力的 ...

  5. 「NOI2017」游戏

    「NOI2017」游戏 题目描述 小 L 计划进行 \(n\) 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 \(A\).\(B\).\ ...

  6. loj #2305. 「NOI2017」游戏

    #2305. 「NOI2017」游戏 题目描述 小 L 计划进行 nnn 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 AAA.BBB. ...

  7. LOJ2305 「NOI2017」游戏

    「NOI2017」游戏 题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行$n$场 ...

  8. [LOJ 2070] 「SDOI2016」平凡的骰子

    [LOJ 2070] 「SDOI2016」平凡的骰子 [题目链接] 链接 [题解] 原题求的是球面面积 可以理解为首先求多面体重心,然后算球面多边形的面积 求重心需要将多面体进行四面体剖分,从而计算出 ...

  9. liberOJ #2033. 「SDOI2016」生成魔咒 后缀数组

    #2033. 「SDOI2016」生成魔咒     题目描述 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1 11.2 22 拼凑起来形成一个魔咒串 [1,2] [1, 2] ...

随机推荐

  1. VS2012快捷键消失

    我也是网上搜的不过我认为挺有效就自己摘录下来了,具体原作者也找不到,所以就下手了,望原谅. 开始菜单 -->所有程序-->Visual Studio 2012文件夹 --> Visu ...

  2. MapReduce的编程思想(1)

    MapReduce的编程思想(1) MapReduce的过程(2) 1. MapReduce采用分而治之的思想,将数据处理拆分为主要的Map(映射)与Reduce(化简)两步,MapReduce操作数 ...

  3. 我们为什么要看《超实用的HTML代码段》

    不知道自己HTML水平如何,不知道HTML5如何进化?看这张图 如果一半以上的你都不会,必须看这本书,阿里一线工程师用代码和功能页面来告诉你每一个技术点. 都会一点,但不知道如何检验自己,看看本书提供 ...

  4. hdu 3466 Proud Merchants 自豪的商人(01背包,微变形)

    题意: 要买一些东西,每件东西有价格和价值,但是买得到的前提是身上的钱要比该东西价格多出一定的量,否则不卖.给出身上的钱和所有东西的3个属性,求最大总价值. 思路: 1)WA思路:与01背包差不多,d ...

  5. connect() to 192.168.30.71:8082 failed (99: Cannot assign requested address) while connecting to upstream, client: 114.80.182.136, server: localhost, request: "GET /home/senior HTTP/1.1", upstream: "

    connect() to 192.168.30.71:8082 failed (99: Cannot assign requested address) while connecting to ups ...

  6. UVALive 3026 Period (KMP算法简介)

    kmp的代码很短,但是不太容易理解,还是先说明一下这个算法过程吧. 朴素的字符串匹配大家都懂,但是效率不高,原因在哪里? 匹配过程没有充分利用已经匹配好的模版的信息,比如说, i是文本串当前字符的下标 ...

  7. ubuntu 18.04下 配置qt opencv的坑

    问题和过程描述: 我按照网上的教程装了qt5.8版本,然后去配置opencv,感觉一切顺利,然后随便写了个 Mat src = imread("xxx") 然后imshow发现编译 ...

  8. JS的闭包、高阶函数、柯里化

    本文原链接:https://cloud.tencent.com/developer/article/1326958 https://cloud.tencent.com/developer/articl ...

  9. 2018.2.14 Java中的哈夫曼编码

    概念 哈夫曼编码(Huffman Coding),又称霍夫曼编码,是一种编码方式,哈夫曼编码是可变字长编码(VLC)的一种.Huffman于1952年提出一种编码方法,该方法完全依据字符出现概率来构造 ...

  10. RestSharp使用备忘

    (1)一般调用: public static List<T> Execute<T>(string resourceUrl, object obj, out int totalN ...