刚开始, 我以为两个点肯定是通过树上最短路径过去的, 无非是在两棵树之间来回切换, 这个可以用倍增 + dp

去维护它。 但是后来又发现, 它可以不通过树上最短路径过去, 我们考虑这样一种情况, 起点在奇树里面, 终点
在偶树里面, 然后这两个点最短路径里面点到对应点的距离都很大, 这种情况下我们就需要从别的地方绕过去, 这样

就不是走树上最短路径了, 但是如果我们将对应点的距离更新成最短距离, 上面这个倍增 + dp的方法就可行了, 所以

我们可以用最短路去更新对应点之间的距离, 将它变成最小值。

#include<bits/stdc++.h>
#define LL long long
#define fi first
#define se second
#define mk make_pair
#define PLL pair<LL, LL>
#define PLI pair<LL, int>
#define PII pair<int, int>
#define SZ(x) ((int)x.size())
#define ull unsigned long long using namespace std; const int N = 3e5 + ;
const int inf = 0x3f3f3f3f;
const LL INF = 0x3f3f3f3f3f3f3f3f;
const int mod = ;
const double eps = 1e-;
const double PI = acos(-); int n, q, depth[N];
LL d[N], gg[N], dp[N][][][];
int f[N][];
vector<pair<int, PLL>> G[N];
vector<PLI> E[N]; void dfs(int u, int fa, PLL dis) {
depth[u] = depth[fa] + ;
if(u > ) {
f[u][] = fa;
dp[u][][][] = min(dis.fi, dis.se + d[u] + d[fa]);
dp[u][][][] = min(dis.se, dis.fi + d[u] + d[fa]);
dp[u][][][] = min(dis.fi + d[fa], dis.se + d[u]);
dp[u][][][] = min(dis.se + d[fa], dis.fi + d[u]);
for(int i = ; i < ; i++) f[u][i] = f[f[u][i - ]][i - ];
for(int i = ; i < ; i++) {
int v = f[u][i - ];
dp[u][][][i] = min(dp[u][][][i - ] + dp[v][][][i - ], dp[u][][][i - ] + dp[v][][][i - ]);
dp[u][][][i] = min(dp[u][][][i - ] + dp[v][][][i - ], dp[u][][][i - ] + dp[v][][][i - ]);
dp[u][][][i] = min(dp[u][][][i - ] + dp[v][][][i - ], dp[u][][][i - ] + dp[v][][][i - ]);
dp[u][][][i] = min(dp[u][][][i - ] + dp[v][][][i - ], dp[u][][][i - ] + dp[v][][][i - ]);
}
}
for(auto& e : G[u]) {
if(e.fi == fa) continue;
dfs(e.fi, u, e.se);
}
} int getLca(int u, int v) {
if(depth[u] < depth[v]) swap(u, v);
for(int i = ; i >= ; i--)
if(depth[f[u][i]] >= depth[v]) u = f[u][i];
if(u == v) return u;
for(int i = ; i >= ; i--)
if(f[u][i] != f[v][i]) u = f[u][i], v = f[v][i];
return f[u][];
} PLL calc(int u, int op1, int v) {
int dis = depth[u] - depth[v];
LL g[], tmp[];
g[op1] = ;
g[op1 ^ ] = d[u];
for(int i = ; i >= ; i--) {
if(dis >> i & ) {
tmp[] = g[], tmp[] = g[];
g[] = min(tmp[] + dp[u][][][i], tmp[] + dp[u][][][i]);
g[] = min(tmp[] + dp[u][][][i], tmp[] + dp[u][][][i]);
u = f[u][i];
}
}
return mk(g[], g[]);
} int main() {
scanf("%d", &n);
for(int i = ; i <= n; i++) scanf("%lld", &d[i]);
for(int i = ; i <= n; i++) {
int u, v; LL w1, w2;
scanf("%d%d%lld%lld", &u, &v, &w1, &w2);
G[u].push_back(mk(v, mk(w1, w2)));
G[v].push_back(mk(u, mk(w1, w2)));
E[u].push_back(mk(w1 + w2, v));
E[v].push_back(mk(w1 + w2, u));
}
priority_queue<PLI, vector<PLI>, greater<PLI> > que;
for(int i = ; i <= n; i++) {
gg[i] = d[i];
que.push(mk(d[i], i));
}
while(!que.empty()) {
int u = que.top().se;
LL val = que.top().fi;
que.pop();
if(val > gg[u]) continue;
for(auto& e : E[u]) {
if(gg[e.se] > val + e.fi) {
gg[e.se] = val + e.fi;
que.push(mk(gg[e.se], e.se));
}
}
}
for(int i = ; i <= n; i++) d[i] = gg[i];
dfs(, , mk(, ));
scanf("%d", &q);
while(q--) {
int u, v;
scanf("%d%d", &u, &v);
int op1 = (u & ) ? : ;
int op2 = (v & ) ? : ;
if(u & ) u = (u + ) >> ;
else u >>= ;
if(v & ) v = (v + ) >> ;
else v >>= ;
int Lca = getLca(u, v);
PLL disu = calc(u, op1, Lca);
PLL disv = calc(v, op2, Lca);
printf("%lld\n", min(disu.fi + disv.fi, disu.se + disv.se));
}
return ;
} /* */

Codeforces 1140G Double Tree 倍增 + dp的更多相关文章

  1. Codeforces 225C Barcode(矩阵上DP)

    题目链接:http://codeforces.com/contest/225/problem/C 题目大意: 给出一个矩阵,只有两种字符'.'和'#',问最少修改多少个点才能让每一列的字符一致,且字符 ...

  2. Codeforces 988F Rain and Umbrellas(DP)

    题目链接:http://codeforces.com/contest/988/problem/F 题目大意: 有三个整数a,n,m,a是终点坐标,给出n个范围(l,r)表示这块区域下雨,m把伞(p,w ...

  3. Problem - D - Codeforces Fix a Tree

    Problem - D - Codeforces  Fix a Tree 看完第一名的代码,顿然醒悟... 我可以把所有单独的点全部当成线,那么只有线和环. 如果全是线的话,直接线的条数-1,便是操作 ...

  4. Codeforces 68D - Half-decay Tree

    题意 有一颗高度为 \(h\) 的完全二叉树(即点数为 \(2^{h+1}-1\) ),有两种操作: add x y 给 \(x\) 点的权值加 \(y\) decay 一次衰变定义为选择一个叶子节点 ...

  5. Codeforces 834D - The Bakery(dp+线段树)

    834D - The Bakery 思路:dp[i][j]表示到第j个数为止分成i段的最大总和值. dp[i][j]=max{dp[i-1][x]+c(x+1,j)(i-1≤x≤j-1)},c(x+1 ...

  6. zoj 3649 lca与倍增dp

    参考:http://www.xuebuyuan.com/609502.html 先说题意: 给出一幅图,求最大生成树,并在这棵树上进行查询操作:给出两个结点编号x和y,求从x到y的路径上,由每个结点的 ...

  7. [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆)

    [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆) 题面 一棵二叉树的所有点的点权都是给定的集合中的一个数. 让你求出1到m中所有权 ...

  8. 洛谷 P1613 跑路 (倍增 + DP + 最短路)

    题目链接:P1613 跑路 题意 给定包含 \(n\) 个点和 \(m\) 条边的有向图,每条边的长度为 \(1\) 千米.每秒钟可以跑 \(2^k\) 千米,问从点 \(1\) 到点 \(n\) 最 ...

  9. 熟练剖分(tree) 树形DP

    熟练剖分(tree) 树形DP 题目描述 题目传送门 分析 我们设\(f[i][j]\)为以\(i\)为根节点的子树中最坏时间复杂度小于等于\(j\)的概率 设\(g[i][j]\)为当前扫到的以\( ...

随机推荐

  1. $Django 前后端之 跨域问题(同源策略) vue项目(axios跨域请求数据)

    1 跨域问题(多个域之间的数据访问) #同源策略(ip port 协议全部相同) #本站的只能请求本站域名的数据 #CORS实现(跨域资源共享) #实现CORS通信的关键是服务器.只要服务器实现了CO ...

  2. $Django Rest Framework-序列化组件

    1 序列化组件 e=serializers.SerializerMethodField()   # 方法名:叫get_字段名, 参数,返回字典  def get_e(self,obj): #obj为b ...

  3. 轻松搞懂elasticsearch概念

      本文主要介绍elasticsearch6.0的一些基本概念,有助于深入理解.研究elasticsearch和elk系统 一图胜千言 elasticsearch与mysql参照来看 添加一条数据 紫 ...

  4. 做了5年的Android,我转Java后台了!

    很多人做Java开发4,5年后,都会感觉自己遇到瓶颈.什么都会又什么都不会,如何改变困境,为什么很多人写了7,8年还是一个码农,工作中太多被动是因为不懂底层原理.公司的工作节奏又比较快,难有机会学习架 ...

  5. Eclipse中设置Java代码格式化

    一.自定义 Java 代码格式化 [Java-Code-Formatting.xml 下载],下载完毕以后,打开 Eclipse 找到如下图界面,点击 Import 导入即可.

  6. list的add()方法与addAll()方法简介

    简单描述:月读别人的代码,发现了一个有意思的东西,list的一个方法,addAll(),然后就去度娘了一下,发现这个还挺有用的. 吐槽一下:为什么自己没发现这个方法呢?因为平时自己写list的时候,基 ...

  7. Question Of AI Model Training

    1 模型训练基本步骤 准备原始数据,定义神经网络结构及前向传播算法 定义loss,选择反向传播优化算法 生成Session,在训练数据进行迭代训练,使loss到达最小 在测试集或者验证集上对准确率进行 ...

  8. Laravel5使用QQ邮箱发送邮件配置

    在.env文件中设置如下MAIL_DRIVER=smtpMAIL_HOST=smtp.qq.comMAIL_PORT=465MAIL_USERNAME=00000000000@qq.comMAIL_P ...

  9. HTML&javaSkcript&CSS&jQuery&ajax(四)

    一.HTML创建响应设计 Responsive Web Design 可以改变尺寸传递网页,对于平板和移动设备是必须的 1.<!DOCTYPE html><html lang=&qu ...

  10. django----常用功能

    request.path_info 获取url地址