zoukankan      html  css  js  c++  java
  • [PA2012]Tax

    Description

    给定一个图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值。起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边权。求 (1)(n) 的最短路。

    Solution

    如果直接在这个图上跑的话,需要记录一下是从哪条边转移过来的,不太行。换一种思路,因为都是在和边打交道,不妨将边抽成点,建一个新图。点权就是原边权,边就是原图的点,边权就是新图两个点的较大值。但是这样的建图是 (m^2) 的。考虑到取 (max) 这个操作具有单调性。将原图一个点所连的边按边权排序后,一种权值对应的其实是一个前缀,所以考虑前缀优化。这样就可以直接跑了。注意建超级源汇点,以及拆边要拆成两条有向边。

    #include<stdio.h>
    #include<algorithm>
    #include<vector>
    #include<queue>
    using namespace std;
    
    typedef long long ll;
    
    inline int read(){
        int x=0,flag=1; char c=getchar();
        while(c<'0'||c>'9'){if(c=='-') flag=0;c=getchar();}
        while(c>='0'&&c<='9'){x=(x<<1)+(x<<3)+c-48;c=getchar();}
        return flag? x:-x;
    }
    
    const int N=1e5+7;
    const int M=4e5+7;
    
    int s,t,n,m;
    
    struct E_{
        int id; ll w;
        E_(int id_=0,ll w_=0):id(id_),w(w_){}
        bool operator <(const E_ &X)const{return w>X.w;}
    };
    
    int head[M],cnt=1;
    struct E{int next,to,w;}e[M<<5];
    
    inline void add(int id,int to,int w){
        e[++cnt]=(E){head[id],to,w};
        head[id]=cnt;
    //    printf("Link %d %d %d
    ",id,to,w);
    }
    
    ll dis[M];
    priority_queue<E_> Q;
    bool vis[M];
    vector<E_> g[N];
    
    int main(){
        n=read(),m=read();
        for(int i=1,u,v,w;i<=m;i++){
            u=read(),v=read(),w=read();
            g[u].push_back(E_(++cnt,w));
            g[v].push_back(E_(++cnt,w));
        }
        s=1,t=cnt+1; cnt=0;
        for(int i=1;i<=n;i++){
            sort(g[i].begin(),g[i].end());
            E_ lt; bool tag=0;
            for(E_ x:g[i]){
                if(tag){
                    add(lt.id,x.id,0);
                    add(x.id,lt.id,lt.w-x.w);
                }
                add(x.id,x.id^1,x.w);
                tag=1,lt=x;
            }
        }
        for(E_ x:g[1]) add(s,x.id,x.w);
        for(E_ x:g[n]) add(x.id^1,t,x.w);
        for(int i=s;i<=t;i++) dis[i]=1ll<<50;
        dis[s]=0,Q.push(E_(s,0));
        while(!Q.empty()){
            E_ t=Q.top(); Q.pop(); int u=t.id;
            if(vis[u]) continue; vis[u]=1;
            for(int i=head[u];i;i=e[i].next){
                int v=e[i].to;
                if(dis[v]>dis[u]+e[i].w){
                    dis[v]=dis[u]+e[i].w;
                    if(!vis[v]) Q.push((E_){v,dis[v]});
                }
            }
        }
        printf("%lld",dis[t]);
    }
    
  • 相关阅读:
    学算法的那些年,吴师兄接触的网站、软件、视频、书籍大揭秘
    阮一峰:CSS Modules 用法教程
    截取url参数
    在dotnet core实现类似crontab的定时任务
    开源一个基于dotnet standard的轻量级的ORM框架-Light.Data
    ABP Vnext使用mysql数据库
    实现ElementUI Dialog宽度响应式变化
    使用Vue Baidu Map对百度地图实现输入框搜索定位
    使用Docker搭建HttpRunnerManager环境
    SpringBoot集成spring aop开发
  • 原文地址:https://www.cnblogs.com/wwlwQWQ/p/15209522.html
Copyright © 2011-2022 走看看