标题效果:一个N积分m无向图边。它可以是路径k右边缘值变0,确定此时1-n最短路径长度。

Sol:我以为我们考虑分层图,图复制k+1部分,每间0~k一层。代表在这个时候已经过去“自由边缘”文章编号。

层与层之间的边权值为0且为单向由上层指向下层。

这样我们以0层的1点做单源最短路径。每一层的n点的距离最小值即为答案。

仅仅只是这种点数为O(K*N),边数为O(K*M),比較慢。

我的做法是,对每一层使用heap-dijkstra算法由本层的原点更新这一层的最短路长度。然后显然能够用O(m)的复杂度推知下一层的初始最短路长度。

这样的做法显然空间和时间上均存在较大优势。

Code:

#include <cstdio>
#include <cstring>
#include <cctype>
#include <iostream>
#include <algorithm>
#include <queue>
#include <stack>
using namespace std; inline int getc() {
static const int L = 1 << 15;
static char buf[L], *S = buf, *T = buf;
if (S == T) {
T = (S = buf) + fread(buf, 1, L, stdin);
if (S == T)
return EOF;
}
return *S++;
}
inline int getint() {
int c;
while(!isdigit(c = getc()));
int tmp = c - '0';
while(isdigit(c = getc()))
tmp = (tmp << 1) + (tmp << 3) + c - '0';
return tmp;
} typedef long long LL; #define N 10010
#define M 50010
int n, m, k;
int head[N], next[M << 1], end[M << 1], len[M << 1];
LL dis[2][N];
bool inpath[N]; queue<int> q; void addedge(int a, int b, int _len) {
static int q = 1;
len[q] = _len;
end[q] = b;
next[q] = head[a];
head[a] = q++;
}
void make(int a, int b, int _len) {
addedge(a, b, _len);
addedge(b, a, _len);
} struct Node {
int lab, dis;
Node(int _lab = 0, int _dis = 0):lab(_lab),dis(_dis){}
bool operator < (const Node &B) const {
return (dis < B.dis) || (dis == B.dis && lab < B.lab);
}
};
struct Heap {
Node a[N];
int top, ch[N];
Heap():top(0){}
void up(int x) {
for(; x != 1; x >>= 1) {
if (a[x] < a[x >> 1]) {
swap(ch[a[x].lab], ch[a[x >> 1].lab]);
swap(a[x], a[x >> 1]);
}
else
break;
}
}
void down(int x) {
int son;
for(; x << 1 <= top; ) {
son=(((x<<1)==top)||(a[x<<1]<a[(x<<1)|1]))?(x<<1):((x<<1)|1);
if (a[son] < a[x]) {
swap(ch[a[son].lab], ch[a[x].lab]);
swap(a[son], a[x]);
x = son;
}
else
break;
}
}
void insert(Node x) {
a[++top] = x;
ch[x.lab] = top;
up(top);
}
Node Min() {
return a[1];
}
void pop() {
a[1] = a[top];
ch[a[top--].lab] = 1;
down(1);
}
void change(int x, int to) {
int ins = ch[x];
a[ins].dis = to;
up(ins);
}
}H; void Dijkstra(bool d) {
H.top = 0;
int i, j;
memset(inpath, 0, sizeof(inpath));
for(i = 1; i <= n; ++i)
H.insert(Node(i, dis[d][i]));
for(i = 1; i <= n; ++i) {
Node tmp = H.Min();
H.pop();
inpath[tmp.lab] = 1;
for(j = head[tmp.lab]; j; j = next[j]) {
if (!inpath[end[j]] && dis[d][end[j]] > dis[d][tmp.lab] + len[j]) {
dis[d][end[j]] = dis[d][tmp.lab] + len[j];
H.change(end[j], dis[d][end[j]]);
}
}
}
} int main() {
n = getint();
m = getint();
k = getint(); int i, j;
int a, b, x;
for(i = 1; i <= m; ++i) {
a = getint();
b = getint();
x = getint();
make(a, b, x);
} int now = 0, last = 1; memset(dis, 0x3f, sizeof(dis));
dis[now][1] = 0;
Dijkstra(now);
LL ans = dis[now][n]; while(k--) {
now ^= 1;
last ^= 1;
for(i = 1; i <= n; ++i)
dis[now][i] = dis[last][i];
for(i = 1; i <= n; ++i)
for(j = head[i]; j; j = next[j])
dis[now][end[j]] = min(dis[now][end[j]], dis[last][i]);
Dijkstra(now);
if (ans == dis[now][n])
break;
ans = dis[now][n];
} printf("%lld", ans); return 0;
}

版权声明:本文博主原创文章。博客,未经同意不得转载。

BZOJ1579 USACO 2009 Feb Gold 3.Revamping Trails Solution的更多相关文章

  1. BZOJ1577 USACO 2009 Feb Gold 1.Fair Shuttle Solution

    权限题,不给传送门啦!在学校OJ上交的.. 有些不开心,又是一道贪心,又是一个高级数据结构的模板,又是看了别人的题解还写崩了QAQ,蒟蒻不需要理由呀. 正经题解: 首先,我们可以由「显然成立法」得出, ...

  2. [USACO 2009 Feb Gold] Fair Shuttle (贪心+优先队列)

    题目大意:有N个站点的轻轨站,有一个容量为C的列车起点在1号站点,终点在N号站点,有K组牛群,每组数量为Mi(1≤Mi≤N),行程起点和终点分别为Si和Ei(1≤Si<Ei≤N).计算最多有多少 ...

  3. [USACO 2018 Feb Gold] Tutorial

    Link: USACO 2018 Feb Gold 传送门 A: $dp[i][j][k]$表示前$i$个中有$j$个0且末位为$k$的最优解 状态数$O(n^3)$ #include <bit ...

  4. USACO 2009 Feb 股票市场 Stock Market

    USACO 2009 Feb 股票市场 Stock Market Description 尽管奶牛们天生谨慎,她们仍然在住房抵押信贷市场中大受打击,现在她们准备在股市 上碰碰运气.贝西开挂了,她知道S ...

  5. 道路翻新 (Revamping Trails, USACO 2009 Feb)

    题意:给定m<=50000的1-n有联通的图,求最多可以使K<=20条边变为0的情况下的最短路是多少.. 思路:简单的分层图最短路,对于每个点拆成K个点.. 然后求一边最短路.. code ...

  6. [USACO 2012 Feb Gold] Cow Coupons【贪心 堆】

    传送门1:http://www.usaco.org/index.php?page=viewproblem2&cpid=118 传送门2:http://www.lydsy.com/JudgeOn ...

  7. 【BZOJ】【3398】【USACO 2009 Feb】Bullcow 牡牛和牝牛

    组合计数/乘法逆元 排列组合求总方案数 这个可以用一个一维的动态规划解决: f[i][0]表示第i头牛是牝牛的方案数 f[i][1]表示第i头牛是牡牛的方案数 则转移为:f[i][0]=f[i-1][ ...

  8. bzoj3939 【USACO 2015 FEB GOLD 】cow hopscotch

    Description 就像人类喜欢玩"跳房子"的游戏,农民约翰的奶牛已经发明了该游戏的一个变种自己玩.由于笨拙的动物体重近一吨打,牛跳房子几乎总是以灾难告终,但这是没有阻止奶牛几 ...

  9. BZOJ1782[USACO 2010 Feb Gold 3.Slowing down]——dfs+treap

    题目描述 每天Farmer John的N头奶牛(1 <= N <= 100000,编号1…N)从粮仓走向他的自己的牧场.牧场构成了一棵树,粮仓在1号牧场.恰好有N-1条道路直接连接着牧场, ...

随机推荐

  1. 十年linux命令总结

    十年linux命令总结 本文链接: http://codingstandards.iteye.com/blog/786653 关于命令类型划分 本表中列出了我穷尽了我所有的记忆整理出来的Linux命令 ...

  2. Android横屏竖屏设置

    Android横竖屏设置: 方法一:onCreate()中 setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE); // ...

  3. 怎样使用jlink一键烧录整个flash Hi3518 a c e Hi3515 Hi3512

    以jlink烧录3515为例: 1\在jlink安装文件夹"C:\Program Files\SEGGER\JLinkARM_V426b"建立批处理文件"HI3515烧写 ...

  4. 探索Oracle数据库升级6 11.2.0.4.3 Upgrade12c(12.1.0.1)

    探索Oracle数据库升级6 11.2.0.4.3 Upgrade12c(12.1.0.1) 一.前言:       Oracle 12c公布距今已经一年有余了,其最大亮点是一个能够插拔的数据库(PD ...

  5. android中listview分页载入数据

    前段时间做的新浪微博项目一直想实现listview分页载入数据,今天最终实现了,哈哈!感觉挺好的,今天又写了个demo给大家分享下. 首先说下listview的优化方案,这也是面试中常考的题目.优化方 ...

  6. ECSHOP如何增加红包序列号字符

    ECSHOP系统线下发放红包时系统生成的红包序列号是在10000的基础上增加四位随机数字.如果当我们要发放大额度红包的时候,这样的序列号规 则难免给人不安全的感觉,万一有无聊的人,蒙几个红包序列号出来 ...

  7. 跨平台移动框架iMAG开发入门

    iMAG是一个非常简洁高效的移动跨平台开发框架,开发一次能够同一时候兼容Android和iOS平台,有点儿Web开发基础就能非常快上手.当前移动端跨平台开发的框架有非常多,但用iMAG另一个优点,就是 ...

  8. 正确Linux新手很实用20命令

     //正确Linux新手很实用20命令 //slwang  2014.4.19 1, ls list directory contents 内容 ls -l     //以详情模式(long li ...

  9. 关于js中window.location.href,location.href,parent.location.href,top.location.href的使用方法

    关于js中"window.location.href"."location.href"."parent.location.href".&qu ...

  10. ECG信号读出,检测QRS,P,T 波(小波去噪,并根据检测),基于BP辨识的神经网络

    这学期的课程选择神经网络.最后的作业处理ECG信号,并利用神经网络识别. 1  ECG引进和阅读ECG信号 1)ECG介绍  详细ECG背景应用就不介绍了,大家能够參考百度 谷歌.仅仅是简单说下ECG ...