zoukankan      html  css  js  c++  java
  • HDU 1217 Arbitrage(Bellman-Ford判断负环+Floyd)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1217

    题目大意:问你是否可以通过转换货币从中获利

    如下面这组样例:

    USDollar 0.5 BritishPound

    BritishPound 10.0 FrenchFranc

    FrenchFranc 0.21 USDollar

    可以通过US->Br->French->US这样转换,把1美元变成1*0.5*10*0.21=1.05美元赚取%5的利润。

    解题思路:其实就相当于bellman-ford里的负环判断,负环的意思是沿着走一圈回到原点后路径会变短了。这里我们可以把路径的减小转化为货币价值的增加,如果最后回到起点(i==V-1)继续更新别的点(i==V),说明起点价值相对原来变大了。也可以用floyd写,查看个点价值是否变大。

     1 #include<iostream>
     2 #include<string>
     3 #include<map>
     4 #include<queue>
     5 using namespace std;
     6 const int N=35;
     7 const int INF=1<<30-1;
     8 struct edge{
     9     int from,to;
    10     double rate;
    11 }eg[N*N];
    12 
    13 int V,E;
    14 double d[N];
    15 
    16 bool bellman_ford(int s){
    17     for(int i=1;i<=V;i++){
    18         d[i]=0;
    19     }
    20     d[s]=1.0;//初始货币为1 
    21     
    22     for(int i=1;i<=V;i++){
    23         for(int j=1;j<=E;j++){
    24             edge e=eg[j];
    25             //这里是判断会不会变大 
    26             if(d[e.to]<d[e.from]*e.rate){
    27                 d[e.to]=d[e.from]*e.rate;
    28                 if(i==V) return true;//想当于d[s]>1.0 
    29             }
    30         }
    31     }
    32     return false; 
    33 }
    34 
    35 int main(){
    36     int cas=0;
    37     while(cin>>V&&V){
    38         map<string,int>mp;
    39         string s;
    40         for(int i=1;i<=V;i++){
    41             cin>>s;
    42             mp[s]=i;
    43         }
    44         cin>>E;
    45         for(int i=1;i<=E;i++){
    46             string s1,s2;
    47             double rate;
    48             cin>>s1>>rate>>s2;
    49             eg[i].from=mp[s1];
    50             eg[i].to=mp[s2];
    51             eg[i].rate=rate;
    52         }
    53         bool flag=false;
    54         //对每种货币都尝试一遍 
    55         for(int i=1;i<=V;i++){
    56             flag=bellman_ford(i);
    57             if(flag){    
    58                 cout<<"Case "<<++cas<<": Yes"<<endl;
    59                 break;
    60             }
    61         }
    62         if(!flag)
    63             cout<<"Case "<<++cas<<": No"<<endl;
    64     }
    65     return 0;
    66 }

     Floyd:

     1 #include<iostream>
     2 #include<string>
     3 #include<map>
     4 using namespace std;
     5 const int N=35;
     6 const int INF=1<<30-1;
     7 
     8 int V,E;
     9 double val[N][N];
    10 
    11 void floyd(){
    12     for(int k=1;k<=V;k++){
    13         for(int i=1;i<=V;i++){
    14             for(int j=1;j<=V;j++){
    15                 if(val[i][j]<val[i][k]*val[k][j])
    16                     val[i][j]=val[i][k]*val[k][j];
    17             }
    18         }
    19     }
    20 }
    21 
    22 int main(){
    23     int cas=0;
    24     while(cin>>V&&V){
    25         map<string,int>mp;
    26         //路径初始化为0,各个点初始化为1 
    27         for(int i=1;i<=V;i++){
    28             for(int j=1;j<=V;j++){
    29                 val[i][j]=(i==j?1:0);
    30             }
    31         }
    32         for(int i=1;i<=V;i++){
    33             string s;
    34             cin>>s;
    35             mp[s]=i;
    36         }
    37         cin>>E;
    38         for(int i=1;i<=E;i++){
    39             string s1,s2;
    40             double trate;
    41             cin>>s1>>trate>>s2;
    42             val[mp[s1]][mp[s2]]=trate;
    43         }
    44         floyd();
    45         bool flag=false;
    46         //每个点都看一遍是否可以获得利益 
    47         for(int i=1;i<=V;i++){
    48             if(val[i][i]>1){
    49                 flag=true;
    50                 break;
    51             }
    52         }
    53         if(flag){
    54             cout<<"Case "<<++cas<<": Yes"<<endl;
    55         }
    56         else{
    57             cout<<"Case "<<++cas<<": No"<<endl;
    58         }
    59     }
    60 }
  • 相关阅读:
    超级钢琴 2010年NOI
    vijos P1375 大整数(高精不熟的一定要做!)
    COGS 445. [HAOI2010]最长公共子序列
    系统升级
    mariabd mysql升级mariadb
    mysql view 视图
    mysql 杂
    mysql主从复制
    DNS迭代查询与递归查询的区别
    Python 中 str 和 repr 的区别
  • 原文地址:https://www.cnblogs.com/fu3638/p/6986712.html
Copyright © 2011-2022 走看看