• <USACO09JAN>气象测量/气象牛The Baric Bovineの思路


    我..莫名其妙搞出来的

    随便搞搞..幻想中的dp结果对了

    我也很迷茫

    Description

    • 为了研究农场的气候,Betsy帮助农夫John做了N(1 <= N <= 100)次气压测量并按顺序记录了结果M_1...M_N(1 <= M_i <= 1,000,000).
      Betsy想找出一部分测量结果来总结整天的气压分布. 她想用K(1 <= K <= N)个数s_j(1 <= s_1 < s_2 < ... < s_K <= N)来概括所有测量结果. 她想限制如下的误差:对于任何测量结果子集,每一个非此子集中的结果都会产生误差.总误差是所有测量结果的误差之和.更明确第说, 对于每一个和所有s_j都不同的i:
        * 如果 i 小于 s_1, 误差是:2 * | M_i - M_(s_1) | 
        * 如果i在s_j和s_(j+1)之间,误差是:| 2 * M_i - Sum(s_j, s_(j+1)) |     注:Sum(x, y) = M_x + M_y; (M_x 和 M_y 之和)
        * 如果i大于s_K,误差为:2 * | M_i - M_(s_K) |
        Besty给了最大允许的误差E (1 <= E <= 1,000,000),找出最小的一部分结果使得误差最多为E.

    Input

    • 第一行: 两个空格分离的数: N 和 E
      第2..N+1行: 第i+1行包含一次测量记录:M_i

    Output

    • 第一行: 两个空格分开的数: 最少能达到误差小于等于E的测量数目 和 使用那个测量数目能达到的最小误差

    Sample Input

    • 4 20

     10 3

     20 40

    Sample Output

    • 2 17

    Hint

    • 【样例说明】

        选择第二和第四次测量结果能达到最小误差17. 第一次结果的误差是2*|10-3| = 14;第三次结果的误差是|2*20 - (3+40)|=3.

     1 #include<cstdio>
     2 #include<cstring>
     3 #include<iostream>
     4 #include<algorithm>
     5 using namespace std;
     6 int n,e;
     7 long long m[110],ero[110][110],dp[110][110];//ero选i和j产生的误差//dp以j结尾的时候,有i个最少测量数目 能达到的最小误差 
     8 long long abss(long long a)
     9 {
    10     if(a<0)return -a;
    11     return a;
    12 }
    13 int main()
    14 {
    15     int i,j,k;
    16     scanf("%d%d",&n,&e);
    17     for(i=1;i<=n;i++)scanf("%lld",&m[i]);    
    18     long long ans=n,anss=999999999;
    19     
    20     for(k=1;k<=n;k++)
    21     {
    22         for(i=k+1;i<=n;i++)
    23             for(j=k+1;j<=i-1;j++) /*枚举题中的i*/ ero[k][i]+=abss(2*m[j]-m[i]-m[k]);
    24             
    25         for(i=1;i<k;i++) ero[k][0]+=2*abss(m[i]-m[k]); //取k时,小于k的i值产生的误差 
    26         for(i=k+1;i<=n;i++) ero[k][n+1]+=2*abss(m[i]-m[k]);// 取k时 大于k的i值产生的误差 
    27     }//关于误差 
    28     
    29     for(i=1;i<=n;i++) 
    30     {
    31         dp[1][i]=ero[i][0]+ero[i][n+1]; 
    32         if((dp[1][i]<=e) && (dp[1][i]<anss))
    33         {
    34             anss=dp[1][i];ans=1;
    35         }
    36     }//处理只取一个数的情况,也就是只取数i的情况 预处理 
    37     
    38     if(ans==1){printf("%lld %lld
    ",ans,anss);return 0;}//如果可以只取一个数emm且满足条件啦 
    39     
    40     for(k=2;k<=n;k++)//最少能达到误差小于等于E的测量数目
    41         for(i=k;i<=n;i++)//枚举结尾的数 
    42         {
    43             dp[k][i]=9999999999;
    44             for(j=k-1;j<i;j++)//枚举上一轮的结尾数 
    45             {
    46                 long long err=ero[j][i]+ero[i][n+1]-ero[j][n+1]; //相比上一轮多的误差 
    47                 dp[k][i]=min(dp[k][i],dp[k-1][j]+err);
    48             }
    49             if((dp[k][i]<=e) && (dp[k][i]<anss) && (k<=ans)) anss=dp[k][i],ans=k;//找最优解emm 
    50         }
    51     printf("%lld %lld
    ",ans,anss);
    52 return 0;
    53 }
    点击查看丑陋の代码&注释
  • 相关阅读:
    print格式化输出(format)
    Python list,tuple,dict,set高级变量常用方法
    K8s的kubectl常用命令
    C/C++中数组与指针的关系探究
    Java中的不可变类
    JAVA 类总结
    c++ string类find总结
    关于看板娘的事儿
    类型转换
    计算机2进制小数点表示法
  • 原文地址:https://www.cnblogs.com/pile8852/p/9294319.html
Copyright © 2020-2023  润新知