题目传送门

  这是一条通往vjudge的神秘通道

  这是一条通往bzoj的神秘通道

题目大意

  如果一条路径走过的边依次为$e_{1}, e_{2}, \cdots , e_{k}$,那么它的长度为$e_{1} + \max (e_{1}, e_{2}) + \max (e_{2}, e_{3}) + \cdots + \max (e_{k - 1}, e_{k}) + e_{k}$,问点$1$到点$n$的最短路。

  显然需要把状态记在最后一条边上。

  然后给一个菊花图,这个做法就gg了。

  因此考虑一些黑科技。

  可以把边看成点,然后考虑如何在辺与边之间快速连边。

  对于一个点的出边,可以把它们按照权值排序,大的向比它略小的连一条权值为0的边,小的向比它略大的连一天权值为它们边权之差的边。

  然后原图中每条边向它的反向边连一条边权相同的边。

  然后再建两个虚点,一个起点,向以1为起点的边连边,边权不变。一个终点,以$n$为终点的边向它连边。

Code

 /**
* bzoj
* Problem#4289
* Accepted
* Time: 3200ms
* Memory: 31292k
*/
#include <bits/stdc++.h>
#ifndef WIN32
#define Auto "%lld"
#else
#define Auto "%I64d"
#endif
using namespace std;
typedef bool boolean; #define ll long long typedef class Edge {
public:
int end, next, w; Edge(int end = , int next = , int w = ):end(end), next(next), w(w) { }
}Edge; typedef class MapManager {
public:
int ce;
int *h;
Edge *es; MapManager() { }
MapManager(int n, int m):ce(-) {
h = new int[(n + )];
es = new Edge[(m + )];
memset(h, -, sizeof(int) * (n + ));
} void addEdge(int u, int v, int w) {
es[++ce] = Edge(v, h[u], w);
h[u] = ce;
// cerr << u << "->" << v << " " << w << endl;
} Edge& operator [] (int pos) const {
return es[pos];
}
}MapManager; int n, m;
Edge *es;
vector<int> *dg;
MapManager g; inline void init() {
scanf("%d%d", &n, &m);
es = new Edge[(m + )];
dg = new vector<int>[(n + )];
for (int i = ; i < m; i++) {
scanf("%d%d%d", &es[i].end, &es[i].next, &es[i].w);
dg[es[i].end].push_back(i << );
dg[es[i].next].push_back(i << | );
}
} boolean cmp(const int& a, const int& b) {
return es[a >> ].w < es[b >> ].w;
} int s, t; inline void build() {
g = MapManager((m << ) + , m << );
s = m << , t = m << | ;
for (int i = ; i <= n; i++) {
sort(dg[i].begin(), dg[i].end(), cmp);
for (int j = ; j < (signed) dg[i].size(); j++) {
int u = (j) ? (dg[i][j - ]) : (-), v = dg[i][j];
g.addEdge(v, u, );
g.addEdge(u, v, es[v >> ].w - es[u >> ].w);
}
} for (int i = ; i < (signed) dg[].size(); i++)
g.addEdge(s, dg[][i], es[dg[][i] >> ].w);
for (int i = ; i < (signed) dg[n].size(); i++)
g.addEdge(dg[n][i] ^ , t, es[dg[n][i] >> ].w); for (int i = , w; i < m; i++) {
w = es[i].w;
g.addEdge(i << , i << | , w);
g.addEdge(i << | , i << , w);
}
} typedef class Node {
public:
int p;
ll dis; Node(int p = , ll dis = ):p(p), dis(dis) { } boolean operator < (Node b) const {
return dis > b.dis;
}
}Node; ll *f;
priority_queue<Node> que;
inline ll dijstra() {
f = new ll[(m << ) + ];
memset(f, 0x3f, sizeof(ll) * ((m << ) + ));
que.push(Node(s, f[s] = ));
while (!que.empty()) {
Node e = que.top();
que.pop();
if (e.dis != f[e.p]) continue;
for (int i = g.h[e.p]; ~i; i = g[i].next) {
Node eu (g[i].end, e.dis + g[i].w);
if (eu.dis < f[eu.p]) {
f[eu.p] = eu.dis;
que.push(eu);
}
}
}
return f[t];
} inline void solve() {
printf(Auto"\n", dijstra());
} int main() {
init();
build();
solve();
return ;
}

bzoj 4289 Tax - 最短路的更多相关文章

  1. bzoj 4289 TAX —— 点边转化

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4289 把边转化成点,同一个原有点相连的边中,边权小的向大的连差值的边,大的向小的连0的边: ...

  2. 「BZOJ 4289」 PA2012 Tax

    「BZOJ 4289」 PA2012 Tax 题目描述 给出一个 \(N\) 个点 \(M\) 条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点 \(1\) 到点 \( ...

  3. BZOJ 4289: PA2012 Tax 差分建图 最短路

    https://www.lydsy.com/JudgeOnline/problem.php?id=4289 https://www.cnblogs.com/clrs97/p/5046933.html  ...

  4. BZOJ 4289: PA2012 Tax(最短路)

    Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 755  Solved: 240[Submit][Status][Discuss] Descriptio ...

  5. ●BZOJ 4289 PA2012 Tax

    ●赘述题目 算了,题目没有重复的必要. 注意理解:对答案造成贡献的是每个点,就是了. 举个栗子: 对于如下数据: 2 1 1 2 1 答案是 2: ●题解 方法:建图(难点)+最短路. 先来几个链接: ...

  6. BZOJ.4289.PA2012 Tax(思路 Dijkstra)

    题目链接 \(Description\) 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价.起点的代价是离开起点的边的边权,终点的代价 ...

  7. bzoj 4289 PA2012 Tax——构图

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4289 可以把一个点上的边按权值排序,然后边权小的向第一个比它大的连差值的边,边权大的向第一个 ...

  8. bzoj 4289: PA2012 Tax

    Description 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价.起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边 ...

  9. 【刷题】BZOJ 4289 PA2012 Tax

    Description 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价.起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边 ...

随机推荐

  1. unity3d-知识汇总

    itween下载 http://www.youkexueyuan.com/exp_show/1147.html 代码修改精灵图片的透明度 UIBp.GetComponent<Image>( ...

  2. laravel架构

    1.Laravel 5.1 中的异常处理器和HTTP异常处理实例教程 http://laravelacademy.org/post/1867.html 2.laravel 集成sentry,sentr ...

  3. schame定义及用处

    一.schame详解 http://www.cnblogs.com/Neo-ds/p/4790413.html 1.先明确一点,SQL Server中模式(schema)这个概念是在2005的版本里才 ...

  4. Unity shader学习之遮罩纹理

    什么是遮罩? 遮罩允许我们可以保护某些区域,使它们奂于某些修改. 例如下面的例子,使用遮罩来控制高光反射. 转载请注明出处:http://www.cnblogs.com/jietian331/p/71 ...

  5. ModuleNotFoundError: No module named '_pydevd_bundle.pydevd_cython' error on debug

    现象:pycharm调试代码出现错误:ModuleNotFoundError: No module named '_pydevd_bundle.pydevd_cython' error on debu ...

  6. [转]sql server 常用脚本(日常查询所需)

    1:查看sql server代理中作业的运行状况的脚本 -- descr : a simple sql script to view sql server jobs run status -- las ...

  7. PyCharm2017破解步骤

    前段时间买了一套python的学习视频,附带一个Pycharm的安装包和注册码,现在注册码被JetBrains封杀了,不得已在网上找了一下破解的教程,在这里记录一下: 先找到破解补丁无需使用注册码,下 ...

  8. win10 常用设置 桌面出来计算机图标,固定桌面摆好的图标设置方法,电脑设备ID方法

    win10 常用设置 桌面出来计算机图标,固定桌面摆好的图标设置方法 桌面右键-->显示设置-->桌面图标设置 电脑设备ID:xxx查看方法:桌面右键-->显示设置-->关于

  9. hdu 1466 计算直线的交点数 递推

    题目描述 平面上有n条直线,且无三线共点,问这些直线能有多少种不同交点数. 比如,如果n=2,则可能的交点数量为0(平行)或者1(不平行). 输入 输入数据包含多个测试实例,每个测试实例占一行,每行包 ...

  10. ORM for Net主流框架汇总

    ORM框架:Object/Relation Mapping(对象/关系 映射)的缩写,易于理解的模型化数据的方法.简单的说就是把数据库的关系型数据类型转换为用对象型程序控制的框架类型. 框架优缺点分析 ...