BZOJ1579 USACO 2009 Feb Gold 3.Revamping Trails Solution

BZOJ1579 USACO 2009 Feb Gold 3.Revamping Trails Solution

大家好,又见面了,我是全栈君,今天给大家准备了Idea注册码。

标题效果:一个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;
}

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

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/117081.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 关于iframe嵌套页面传值

    关于iframe嵌套页面传值2.在嵌套的js文件里面直接location.search就可以拿到通过url拿过来的值,在发送的url处写个?进行判断到底传过来的是什么值。1.我这边为vue的页面,直接在static里面将页面放进去,通过url进行传值。…

    2025年5月26日
    0
  • java实现第八届蓝桥杯数位和

    java实现第八届蓝桥杯数位和数位和题目描述数学家高斯很小的时候就天分过人。一次老师指定的算数题目是:1+2+…+100。高斯立即做出答案:5050!这次你的任务是类似的。但并非是把一个个的数字加起来,而是对该数字的每一个数位作累加。这样从1加到100的“和”是:901从10加到15是:21,也就是:1+0+1+1+1+2+1+3+1+4+1+5,这个口算都可以出结果的。按这样的“加法”,从1加到1000是…

    2022年6月15日
    35
  • git如何查看分支是哪个分支创建的_git checkout -b 分支名

    git如何查看分支是哪个分支创建的_git checkout -b 分支名如何查看分支gitbranch-vvgit如何切换分支gitcheckoutbranch_name查看远程分支gitbranch-a查看本地分支gitbranch创建分支gitbranchtest切换分支到testgitcheckouttest删除本地分支gitbranch-dxxxxx查看本地和远程分支-a前面带*号的代表你当前工作目录所处的分支gitbranch-a…

    2022年8月22日
    18
  • navicat 15zuixin激活码_通用破解码

    navicat 15zuixin激活码_通用破解码,https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月16日
    45
  • centos7上安装kafka教程「建议收藏」

    centos7上安装kafka教程「建议收藏」一、centos7安装java8去oracle官网下载:https://www.oracle.com/cn/java/technologies/javase/javase-jdk8-downloads.htmltar-zxvfjdk-8u281-linux-x64.tar.gzmkdir/usr/javamvjdk1.8.0_281/usr/java环境变量配置:打开/etc/profile配置文件vi/etc/profile在配置文件末尾添加如下配置:exportJAV

    2022年10月9日
    1
  • TOF相机基本知识

    TOF相机基本知识TOF是Timeofflight的简写,直译为飞行时间的意思。所谓飞行时间法3D成像,是通过给目标连续发送光脉冲,然后利用传感器接收从物体返回的光,通过探测光脉冲的飞行时间来得到目标物的距离。TOF的深度精度不随距离改变而变化,基本能稳定在cm级。1TOF相机的特点:相对二维图像,可通过距离信息获取物体之间更加丰富的位置关系,即区分前景与后景深度信息依旧可以完成对目标图像

    2022年5月9日
    92

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号