BZOJ 4004: [JLOI2015]装备购买 高斯消元解线性基

BZOJ严重卡精,要加 $long$  $double$ 才能过. 

题意:求权和最小的极大线性无关组.
之前那个方法解的线性基都是基于二进制拆位的,这次不行,现在要求一个适用范围更广的方法.
考虑贪心:将向量组按照代价从小到大排序,依次考虑加入每一组向量,如果能被表示出来就加,表示不出来就不加.
你可能会举出一个反例:按照权值从小到大排序后要加入向量 $x,$ 但是后面有若干向量 $a,b,c,d...$ 能表示出 $x,$ 而 $x$ 却表示不出它们,你可能会说最优解法是加入后面那几个,而不加入 $x.$
然而,你可以列一个等式,就是 $a\times x_{1}+b\times x_{2}+c\times x_{3}....=x,$ 将 $x$ 移到左面,随便一个向量移到右面,变成 $a\times x_{1}+b\times x_{2}-x....=c\times x_{3}.$
而 $x$ 的代价显然要小于 $c,$ 所以我们上述的贪心策略是正确的.

#include <cstdio> 
#include <algorithm>    
#define N 600  
#define eps 1e-5
#define setIO(s) freopen(s".in","r",stdin) 
using namespace std; 
struct Node 
{ 
    int w; 
    long double a[N];      
}p[N];  
bool cmp(Node a,Node b) 
{
    return a.w<b.w; 
}
int n,m;  
int mark[N];   
int main()  
{ 
    int i,j; 
    // setIO("input"); 
    scanf("%d%d",&n,&m); 
    for(i=1;i<=n;++i) 
    {
        for(j=1;j<=m;++j) 
        {
            double c; 
            scanf("%lf",&c);     
            p[i].a[j]=(long double) c;    
        }
    } 
    for(i=1;i<=n;++i) scanf("%d",&p[i].w);    
    sort(p+1,p+1+n,cmp);     
    int ans=0, cnt=0, k; 
    for(i=1;i<=n;++i) 
    {
        for(j=1;j<=m;++j) 
        { 
            if(p[i].a[j]>=-eps&&p[i].a[j]<=eps) continue;   
            if(!mark[j]) 
            {
                mark[j]=i,ans+=p[i].w,++cnt; 
                break; 
            } 
            else 
            {
                long double div=(long double)p[i].a[j]/p[mark[j]].a[j];   
                for(k=j;k<=m;++k) 
                { 
                    p[i].a[k]=(long double)(p[i].a[k]-(long double)div*p[mark[j]].a[k]);     
                }
            }
        }
    }
    printf("%d %d\n",cnt,ans);   
    return 0; 
}

  

posted @ 2019-09-17 19:29  EM-LGH  阅读(175)  评论(0编辑  收藏  举报