[PA2012] Tax

传送门:>Here<

题意:给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价。起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边权N<=100000 M<=200000

解题思路

不免要吐槽一下这题的数据,久调一下午无果与标程对拍没有任何差错不知道为什么就是WA 既然极限数据已经和标程拍上了那么权当出了吧……

不过还是一道好题

首先考虑这道题暴力的做法——将每条边作为新图的点,然后枚举原图的点,遍历一遍这个点相邻的所有边,按照大小打擂在新图中连边,但是这样边的数量多达$M^2$

可以考虑优化边的数量,用到差分的思想——以其中一条边作为基准,往上走要加,往下走不加。作为基准的这一条边也就是当前路径的入边,至于出边,只需要沿着差分的边走就可以了。于是我们所需要做的就是将每个点相邻的所有边排序,并且相邻的连边——大的往小的权值为0,小的往大的权值为差值。并且对于每一条边,它的反向边应当与它连一条权值为其本身的边,作为基准嘛

Code

/*By DennyQi 2018.8.11*/
#include <cstdio>
#include <queue>
#include <cstring>
#include <algorithm>
#define  r  read()
#define  lr lread()
#define  Max(a,b)  (((a)>(b)) ? (a) : (b))
#define  Min(a,b)  (((a)<(b)) ? (a) : (b))
using namespace std;
typedef long long ll;
const int MAXN = 2000010;
const int MAXM = 2000010;
const int INF = 1e18;
inline int read(){
    int x = 0; int w = 1; register int c = getchar();
    while(c ^ ‘-‘ && (c < ‘0‘ || c > ‘9‘)) c = getchar();
    if(c == ‘-‘) w = -1, c = getchar();
    while(c >= ‘0‘ && c <= ‘9‘) x = (x << 3) + (x << 1) + c - ‘0‘, c = getchar(); return x * w;
}
inline int lread(){
    ll x = 0; int w = 1; register int c = getchar();
    while(c ^ ‘-‘ && (c < ‘0‘ || c > ‘9‘)) c = getchar();
    if(c == ‘-‘) w = -1, c = getchar();
    while(c >= ‘0‘ && c <= ‘9‘) x = (x << 3) + (x << 1) + c - ‘0‘, c = getchar(); return x * w;
}
struct Edge{
    ll len;
    int idx;
}e[MAXM];
struct Dij{
    ll w;
    int idx;
};
inline bool operator < (const Dij& a, const Dij& b){
    return a.w > b.w;
}
int N,M,x,y,S,T,v,top;
int pfirst[MAXM],pnxt[MAXM],pto[MAXM],pnum_edge=-1;
ll pcost[MAXM],cost[MAXM],z;
int first[MAXM],nxt[MAXM],to[MAXM],num_edge=-1;
ll d[MAXM];
bool vis[MAXM];
priority_queue <Dij> q;
inline bool comp(const Edge& a, const Edge& b){
    return a.len < b.len;
}
inline void add(int u, int v, int w){
//    printf("%lld->%lld(%lld)\n",u,v,w);
    to[++num_edge] = v;
    cost[num_edge] = w;
    nxt[num_edge] = first[u];
    first[u] = num_edge;
}
inline void padd(int u, int v, int w){
    pto[++pnum_edge] = v;
//    printf("num(%lld): %lld->%lld(%lld)\n",pnum_edge,u,v,w);
    pcost[pnum_edge] = w;
    pnxt[pnum_edge] = pfirst[u];
    pfirst[u] = pnum_edge;
    if(u == 1){
        add(S, pnum_edge, w);
    }
    if(v == N){
        add(pnum_edge, T, w);
    }
}
inline void Dijkstra(int s){
    for(int i = 0; i <= T; ++i) d[i] = INF;
    d[s] = 0;
    q.push((Dij){0,s});
    ll u,v;
    while(!q.empty()){
        u = q.top().idx; q.pop();
        if(vis[u]) continue;
        vis[u] = 1;
        for(int i = first[u]; i != -1; i = nxt[i]){
            v = to[i];
            if(d[u] + cost[i] < d[v]){
                d[v] = d[u] + cost[i];
                q.push((Dij){d[v],v});
            }
        }
    }
}
int main(){
//    freopen(".in","r",stdin);
//    freopen("qxz.out","w",stdout);
    N = r, M = r;
    memset(pfirst,-1,sizeof(pfirst));
    memset(first,-1,sizeof(first));
    S = M*2+1;
    T = M*2+2;
//    printf("S = %lld  T = %lld\n",S,T);
    for(int i = 1; i <= M; ++i){
        x = r, y = r, z = lr;
        padd(x, y, z);
        padd(y, x, z);
    }
    int v;
    for(int x = 2; x < N; ++x){
        top = 0;
        for(int i = pfirst[x]; i != -1; i = pnxt[i]){
            e[++top] = (Edge){pcost[i], i};
        }
        sort(e+1,e+top+1,comp);
        for(int i = 1; i <= top; ++i){
            if(i < top){
                add(e[i].idx, e[i+1].idx, e[i+1].len-e[i].len);
            }
            if(i > 1){
                add(e[i].idx, e[i-1].idx, 0);
            }
            add(e[i].idx^1, e[i].idx, e[i].len);
        }
    }
    Dijkstra(S);
    printf("%lld", d[T]);
    return 0;
}

原文地址:https://www.cnblogs.com/qixingzhi/p/9463737.html

时间: 2024-08-05 11:15:07

[PA2012] Tax的相关文章

BZOJ4289 : PA2012 Tax

一个直观的想法是把每条边拆成两条有向边,同时每条有向边是新图中的一个点.对于两条边a->b与b->c,两点之间连有向边,费用为两条边费用的最大值.然后新建源点S与汇点T,由S向所有起点为1的边连边,T接受所有终点为n的边,那么答案就是S到T的最短路. 这样子的边数为$O(m^2)$,不能承受. 考虑枚举中转点x,将所有与它有关的边按费用从小到大排序.对于每条边,从以x为终点的点向以x为起点的点连边,费用为该边的费用.从以x为起点的点向下一条边连边,费用为两条边费用的差值,向上一条边连边,费用为

●BZOJ 4289 PA2012 Tax

●赘述题目 算了,题目没有重复的必要. 注意理解:对答案造成贡献的是每个点,就是了. 举个栗子: 对于如下数据: 2 1 1 2 1 答案是 2: ●题解 方法:建图(难点)+最短路. 先来几个链接:(他们为我解题提供了思路,但有些部分看得我有点mengbi) ●http://blog.csdn.net/pure_w/article/details/55060079 ●http://www.cnblogs.com/clrs97/p/5046933.html 建图: 1.把原图的双向边拆成两条单向

BZOJ.4289.PA2012 Tax(思路 Dijkstra)

题目链接 \(Description\) 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价.起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边权. \(Solution\) 最直接的方法是把每条边作为一个点,对于连接同一个点的两条边连一条新边,最后把连接1和n的点分别连S.T,跑最短路 但这样边数是O(m^2)的 对于路径上相邻两条边\((i,j,v1)\)和\((j,k,v2)\),v1<v2,考虑如何构图把v1比v2

BZOJ 4289: PA2012 Tax

Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 755  Solved: 240[Submit][Status][Discuss] Description 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价.起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边权 N<=100000 M<=200000 Input Output Sample Input 4 5 1 2 5 1

【Bzoj4289】PA2012 Tax(Dijkstra+技巧建图)

Description 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价.起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边权 N<=100000 M<=200000 Solution 这题关键在于化边为点,把无向边拆成2条有向边 考虑最直白的一种建图方法,对于每一个点u,它的每一条入边向所有出边连边 但这样边数太多了,最坏是\(M^2\)条边,不可行 考虑用差值来建图,每条出边向第一条比它大的出边连一条权值为权差值的

bzoj4289 PA2012 Tax——点边转化

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4289 好巧妙的转化!感觉自己难以想出来... 参考了博客:https://blog.csdn.net/reverie_mjp/article/details/52134142 把边变成点,相互之间连边: 原图上由一个点连接的许多边之间应该通过连新边达到题目要求的取较大值的目的: 做法就是把一个原图点的关联边排序,然后较小的边向较大的边连边权为差值的新边,较大的边连回去边权为0的新边: 那么

【BZOJ-4289】Tax 最短路 + 技巧建图

4289: PA2012 Tax Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 168  Solved: 69[Submit][Status][Discuss] Description 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点1到点N的最小代价.起点的代价是离开起点的边的边权,终点的代价是进入终点的边的边权 N<=100000 M<=200000 Input Output Sample I

省选之前的未完成的计划(截至到省选)

PLAN OF THE COMING HEOI good problems:-bzoj4823:[Cqoi2017]老C的方块 [*]-bzoj3171:[Tjoi2013]循环格 [*]-bzoj4200:[Noi2015]小园丁与老司机 [*]-bzoj1061:[Noi2008]志愿者招募 [*]-bzoj3600:没有人的算术 [*]-bzoj2806:[Ctsc2012]Cheat [*]-bzoj2219:数论之神 [*]-bzoj2595:[Wc2008]游览计划 [*]-bzoj

【PA2012】【BZOJ4289】Tax

Description 给出一个N个点M条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值.求从起点1到点N的最小代价. 起点的代价是离开起点的边的边权.终点的代价是进入终点的边的边权 N<=100000 M<=200000 Input Output Sample Input 4 5 1 2 5 1 3 2 2 3 1 2 4 4 3 4 8 Sample Output 12 HINT Source 一眼能看出是最短路变种.. 可是我仅仅会n^2建图T_T n^2建图就是边