题意:给出一个有向图,问求一个回路,使得回路上的点权之和/边权之和最大。
这题主要是分析出如何确定ans值。我们将(a1*x1+a2*x2+..+an*xn)/(b1*x1+b2*x2+..+bn*xn)=L,转换为:x1*(a1-b1*L)+x2*(a2-b2*L)+...xn*(an-bn*L)=0
则每次枚举L的值,spfa中边权值为len[]*L-a[],若存在负环回路(即一个点访问次数超过n次)则表示L的值小了,增大L值;反之减小L值.
代码:
#include<iostream> #include<cstring> #include<cstdio> #define MAXN 1052 #define inf 10000000 using namespace std; struct Edge{ int u,len,next; }edge[5*MAXN]; int temp,n,m; int head[MAXN],a[MAXN]; void addEdge(int u,int v,int c) { edge[temp].u=v; edge[temp].len=c; edge[temp].next=head[u]; head[u]=temp; temp++; } double dis[MAXN]; bool vis[MAXN]; int num[MAXN]; int que[MAXN*MAXN]; bool spfa(double mid) { double val; for(int i=0;i<=n;i++) { dis[i]=inf; vis[i]=false; num[i]=0; } dis[1]=0; int headt,tail; headt=0;tail=0; que[tail++]=1; vis[1]=true; num[1]++; while(headt!=tail) { int v=que[headt]; headt++; vis[v]=false; for(int i=head[v];i!=-1;i=edge[i].next) { int u=edge[i].u; val=mid*edge[i].len-a[u]; if(dis[v]+val<dis[u]) { dis[u]=dis[v]+val; if(!vis[u]) { que[tail++]=u; vis[u]=true; num[u]++; if(num[u]>=n) { return false; } } } } } return true; } int main() { while(scanf("%d%d",&n,&m)!=EOF) { temp=0; for(int i=1;i<=n;i++) { scanf("%d",&a[i]); } memset(head,-1,sizeof(head)); for(int i=1;i<=m;i++) { int v,w,c; scanf("%d%d%d",&v,&w,&c); addEdge(v,w,c); } double mid=0;double ans=0; double l=0;double r=2000; while(r-l>=0.001) { mid=(l+r)/2.0; if(spfa(mid)) { r=mid; //减小ans; } else { ans=mid; l=mid; } } printf("%.2lf ",ans); } return 0; }