• CF 351A


    http://codeforces.com/problemset/problem/351/C

    题意:有2*n个浮点数a1,a2,a3...a2*n,把他们分成n队,对于每对<A,B>,对A做floor() 操作,对B 做ceil()操作。生成b1...b2*n, 求|(b1+b2+...+b2*n)-(a1+a2+a3...+a2*n)|的最小值。

    对于每个数ai,对他们做floor()的cost是up()=ai-floor(ai) ,做ceil()的cost是down()=ceil(ai)-ai

    设f[i][j]表示前i个节点有j个做ceil()操作(i-j个floor())的最优值

      f[i][j]=min{f[i-1][j-1]+up(a[i]),   对ai做up()的代价

            f[i-1][j]+down(a[i])}   对ai做down()的代价、

      当i == j时注意f[i-1][j]不存在这种情况.

       这里的min{}里面比较的是绝对值。、

    临界值f[0][0]=0; f[0][1]=0;目标答案|f[2*n][n]|。

    #include <stdio.h>
    #include <string.h>
    #include <math.h>
    #define N 2001
    double min(double a,double b)
    {
        return fabs(a)>fabs(b)? b:a;
    }
    double up(double x)
    {
        return ceil(x)-x;
    }
    double down(double x)
    {
        return x-floor(x);
    }
    double a[2*N+5];
    double f[2*N+5][N+5];
    int main()
    {
        int i,j,m,n;
        while (scanf("%d",&n)!=EOF)
        {
            for (i=1;i <=2*n; i++)
                scanf("%lf",&a[i]);
            f[0][0]=0;
            f[0][1]=0;
           double tmp;
            for (i=1;i<=2*n; i++)
                for (j=0; j<= n && j<=i;j++)
                {
                    tmp=400000000;
                    if (i>j) tmp=f[i-1][j]+down(a[i]);
                    if (j>0)
                        tmp=min(tmp,f[i-1][j-1]-up(a[i]));
                        f[i][j]=tmp;
                }
    
                printf("%0.3lf
    ",fabs(f[2*n][n]));
        }
        return 0;
    }

    好吧,其实仔细看看就是一个背包。

      我们可以把i 这个维度,在空间上优化点。跟0/1背包似的。

         f[j]=min{f[j-1]+up(a[i]),f[j]+down(a[i])}

    不过j循环得downto了,因为f[j-1]在迭代前,还应该是f[i-1][j-1]这个状态,而不是f[i][j-1]了。

    #include <stdio.h>
    #include <string.h>
    #include <math.h>
    #define N 2001
    //#define min(a,b) a>b?b:a
    double min(double a,double b)
    {
        return fabs(a)>fabs(b)? b:a;
    }
    double up(double x)
    {
        return ceil(x)-x;
    }
    double down(double x)
    {
        return x-floor(x);
    }
    double a[2*N+5];
    double f[N+5];
    int main()
    {
        int i,j,m,n;
        while (scanf("%d",&n)!=EOF)
        {
            for (i=1;i <=2*n; i++)
                scanf("%lf",&a[i]);
            f[0]=0; f[1]=0;
           double tmp;
            for (i=1;i<=2*n; i++)
                for (j=min(i,n);j>=0;j--)
                {
                    tmp=400000000;
                    if (i>j) tmp=f[j]+down(a[i]);
                    if (j>0)
                        tmp=min(tmp,f[j-1]-up(a[i]));
                        f[j]=tmp;
                    //f[i][j]=min(f[i-1][j-1]+up(a[i]),f[i-1][j]+down(a[i]));
                }
                printf("%0.3lf
    ",fabs(f[n]));
        }
        return 0;
    }
  • 相关阅读:
    Javascript&Html-系统对话框
    Javascript&Html-延迟调用和间歇调用
    Javascript&Html-弹出窗口的屏蔽程序
    iPhone屏幕旋转
    iPhone深度学习-ARM
    xCode控制台的使用-应用闪退原因跟踪
    IOS-内存检测以及优化
    Javascript-Array
    http与https的区别
    Nginx:处理HTTP请求
  • 原文地址:https://www.cnblogs.com/six-god/p/3353608.html
Copyright © 2020-2023  润新知