zoukankan      html  css  js  c++  java
  • POJ 1860(spfa)

    http://poj.org/problem?id=1860

    题意:汇率转换,与之前的2240有点类似,不同的是那个题它去换钱的时候,是不需要手续费的,这个题是需要手续费的,这是个很大的不同。

    思路:还是转换成为最短路的问题,主要的困难也就是关于它的松弛方程。dist [edge[i].v] < (dist[ tmp ] - edge[ i ].r) * edge[ i ].c  。这个就是松弛方程,当它的钱的数目增多的时候松弛。

     1 #include <stdio.h>
     2 #include <string.h>
     3 #include <queue>
     4 
     5 using namespace std;
     6 
     7 
     8 int m,n,p,head[ 10005 ],pos,num[ 10005 ];
     9 bool vis[ 10005 ];
    10 double val,dist[ 10005 ];
    11 
    12 struct note{
    13     int v,next;
    14     double r,c;
    15 }edge[ 10005 ];
    16 
    17 void add(int x,int v,double r,double c)
    18 {
    19     edge[ pos ].v = v;
    20     edge[ pos ].r = r;
    21     edge[ pos ].c = c;
    22     edge[ pos ].next = head[ x ];
    23     head[ x ] = pos ++;
    24 }
    25 
    26 bool spfa()
    27 {
    28     queue<int >s;
    29     s.push(p);
    30     dist[ p ] = val;
    31     vis[ p ] = true;
    32     num[ p ] ++;
    33     while(!s.empty())
    34     {
    35         int tmp = s.front();
    36         s.pop();
    37         vis[ tmp ] = false;
    38         for( int i = head[ tmp ] ; i != -1 ; i = edge[ i ].next)
    39         {
    40             if( dist[ edge[ i ].v ] < (dist[ tmp ] - edge[ i ].c ) * edge[ i ].r )
    41             {
    42                 dist[ edge[ i ].v ] = (dist[ tmp ] - edge[ i ].c ) * edge[ i ].r;
    43                 if( !vis[ edge[ i ].v ] )
    44                 {
    45                     s.push( edge[ i ].v );
    46                     vis[ edge[ i ].v ] = true;
    47                     num[ edge[ i ].v ] ++;     //可能构成正权回路,这个是用来判断的。但次数比m次要大的时候,就肯定是增多的。可以直接return 。
    48                     if( num[edge[ i ].v] > m)
    49                         return true;
    50                 }
    51             }
    52         }
    53     }
    54     if( dist[ p ] > val ) return true;   //如果之后的钱比之前的多,也return true.
    55     return false;
    56 }
    57 
    58 int main()
    59 {
    60    // freopen("in.txt","r",stdin);
    61     int a,b;
    62     double r1,c1,r2,c2;
    63     pos = 1;
    64     memset( head , -1 , sizeof( head ) );
    65     memset( dist , 0 , sizeof( dist ) );
    66     memset( vis , false ,sizeof( vis ) );
    67     memset( num , 0 , sizeof( num ) );
    68     scanf("%d%d%d%lf",&m,&n,&p,&val);
    69     for( int i = 1 ; i <= n ; i++ )
    70     {
    71         scanf("%d%d%lf%lf%lf%lf",&a,&b,&r1,&c1,&r2,&c2);
    72         add(a,b,r1,c1);
    73         add(b,a,r2,c2);
    74     }
    75     if(spfa()) printf("YES
    ");
    76     else printf("NO
    ");
    77     return 0;
    78 }
  • 相关阅读:
    java方式实现堆排序
    java方式实现归并排序
    用java方式实现快速排序
    Linux中crontab定时任务
    TCP/IP网络协议初识
    github设置添加ssh
    IDM下载工具使用
    Java程序在内存中运行详解
    GitHub的高级搜索方式
    深入理解JavaScript中的堆与栈 、浅拷贝与深拷贝
  • 原文地址:https://www.cnblogs.com/Tree-dream/p/5744831.html
Copyright © 2011-2022 走看看