zoukankan      html  css  js  c++  java
  • codeforces 743E

    题解:

    考虑这个每个数字少个数是可二分的(答案关于个数单调)

    然后我们dp

    dp[i][S]表示到第i个数,已选完的数字集合为S

    转移是贪心取连续一段

    预处理转移右端点可以降下复杂度

     1 #include<bits/stdc++.h>
     2 #define maxn 1005
     3 using namespace std;
     4 int n,m;
     5 int a[maxn],c[9];
     6 int dp[maxn][260];
     7 int f[9][maxn][130];
     8 void init()
     9 {
    10     for(int x=1;x<=8;++x)
    11     {
    12         for(int l=0;l<=n;++l)
    13         {
    14             int cnt=0;
    15             f[x][l][0]=l;
    16             for(int r=l+1;r<=n;++r)
    17             {
    18                 if(a[r]==x)
    19                 {
    20                     ++cnt;
    21                     f[x][l][cnt]=r;
    22                 }
    23             }
    24             for(int i=cnt+1;i<=m;++i)f[x][l][i]=n+1;
    25         }
    26     }
    27 }
    28 int ans=0;
    29 bool check(int L)
    30 {
    31     memset(dp,128,sizeof(dp));
    32     dp[0][0]=0;
    33     for(int i=0;i<n;++i)
    34     {
    35         for(int S=0;S<(1<<8);++S)if(dp[i][S]>=0)
    36         {
    37             dp[i+1][S]=max(dp[i+1][S],dp[i][S]);
    38             for(int k=1;k<=8;++k)if(!(S&(1<<(k-1))))
    39             {
    40                 int sta=S|(1<<(k-1));
    41                 dp[f[k][i][L]][sta]=max(dp[f[k][i][L]][sta],dp[i][S]+L);
    42                 dp[f[k][i][L+1]][sta]=max(dp[f[k][i][L+1]][sta],dp[i][S]+L+1);
    43             }
    44         }
    45     }
    46     ans=max(ans,dp[n][(1<<8)-1]);
    47     if(dp[n][(1<<8)-1]>=0)return 1;
    48     else return 0;
    49 }
    50 int main()
    51 {
    52     scanf("%d",&n);
    53     for(int i=1;i<=n;++i)scanf("%d",&a[i]),c[a[i]]++;
    54     m=n;
    55     for(int i=1;i<=8;++i)m=min(m,c[i]);
    56     m++;
    57     init();
    58     int l=0,r=m-1;
    59     while(l<=r)
    60     {
    61         int mid=(l+r)>>1;
    62         if(check(mid))l=mid+1;
    63         else r=mid-1;
    64     }
    65     printf("%d
    ",ans);
    66     return 0;
    67 }
    View Code
  • 相关阅读:
    MFC调用C动态库函数-----待补充
    硬盘知识总结:
    Android 四:区分刷机与root
    总结:Linux系统启动流程
    Android 三:手机adb 命令解锁
    UVa11136 Hoax or what
    UVa11988 Broken Keyboard (a.k.a. Beiju Text)
    UVa11280 Flying to Fredericton
    UVa10269 Adventure of Super Mario
    UVa12589 Learning Vector
  • 原文地址:https://www.cnblogs.com/uuzlove/p/10625009.html
Copyright © 2011-2022 走看看