【挖坑記】JZOJ 4735 最小圈

題目大意

對於一張有向圖,要你求圖中所有圈的平均值中,最小的是多少,即若一個圈經過k個節點,那麼一個圈的平均值爲圈上k條邊權的和除以k,現要求其中的最小值。
n<=3000,m<=10000
abs(Wi,j)<=10^5
時間限制 1s
空間限制 256M

解題思路

0/1分數規劃,二分答案mid,把邊權減少mid,跑一遍最短路看是否能找到圈。

#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 3005
#define fr(i,a,b) for(i=a;i<=b;i++)
using namespace std;
typedef long long ll;
const int ding=1e5;
const double eps=1e-7;

struct poi
{
    int x;
    double y,z;
    poi *nex;
} *a[maxn];
int i,n,m,x,y;
double z,mn,mx,l,r,mid,tot,dis[maxn];
bool cai,kan[maxn];
void link(int x,int y,double z)
{
    poi *p=new poi;
    p->x=y;
    p->y=z;
    p->nex=a[x];
    a[x]=p;
    return;
}
void dfs(int x)
{
    kan[x]=1;
    poi *p=new poi;
    for(p=a[x];p;p=p->nex)
        if (dis[p->x]>dis[x]+p->y-mid)
        {
            dis[p->x]=dis[x]+p->y-mid;
            if (!kan[p->x]) dfs(p->x);
            else cai=1;
            if (cai) return;
        }
    kan[x]=0;
    return;
}
bool check(double x)
{
    cai=0;
    fr(i,1,n)
    {
        memset(kan,0,sizeof(kan));
        memset(dis,0,sizeof(dis));
        dfs(i);
        if (cai) return 1;
    }
    return 0;
}
int main()
{
    scanf("%d%d",&n,&m);
    mn=ding,mx=-ding;
    fr(i,1,m)
    {
        scanf("%d%d%lf",&x,&y,&z);
        link(x,y,z);
        mn=min(mn,z),mx=max(mx,z);
    }
    l=mn,r=mx;
    while (r-l>eps)
    {
        mid=(l+r)/2;
        if (check(mid)) r=mid;
        else l=mid;
    }
    printf("%.6f\n",l);
    return 0;
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章