zoukankan      html  css  js  c++  java
  • HDU4057 Rescue the Rabbit(AC自动机+状压DP)

    题目大概是给几个DNA片段以及它们各自的权值,如果一个DNA包含某个片段那么它的价值就加上这个片段的权值,同时包含多个相同DNA片段也只加一次,问长度l的DNA可能的最大价值。

    HDU2825大同小异。

    • dp[i][j][S]表示长度i(自动机转移i步)、后缀状态为自动机第j个结点、包含的DNA片段为集合S 的DNA最大价值
    • dp[0][0][0]=0
    • 我为人人转移,从dp[i][j][S]向ATCG四个方向更新dp[i+1][j'][S']
    1. 注意的是要用滚动数组,不然要开上百兆数组。
    2. 用滚动数组要注意初始化。
    3. 转移过程有些计算可以预处理出来,存在数组里,虽然不预处理应该也不会TLE,题目时间给了10000ms。
     1 #include<cstdio>
     2 #include<cstring>
     3 #include<queue>
     4 using namespace std;
     5 #define INF (1<<30)
     6 int tn,ch[1111][4],fail[1111],flag[1111];
     7 int idx[128];
     8 void insert(char *s,int k){
     9     int x=0;
    10     for(int i=0; s[i]; ++i){
    11         int y=idx[s[i]];
    12         if(ch[x][y]==0) ch[x][y]=++tn;
    13         x=ch[x][y];
    14     }
    15     flag[x]|=1<<k;
    16 }
    17 void init(){
    18     memset(fail,0,sizeof(fail));
    19     queue<int> que;
    20     for(int i=0; i<4; ++i){
    21         if(ch[0][i]) que.push(ch[0][i]);
    22     }
    23     while(!que.empty()){
    24         int x=que.front(); que.pop();
    25         for(int i=0; i<4; ++i){
    26             if(ch[x][i]) que.push(ch[x][i]),fail[ch[x][i]]=ch[fail[x]][i],flag[ch[x][i]]|=flag[ch[fail[x]][i]];
    27             else ch[x][i]=ch[fail[x]][i];
    28         }
    29     }
    30 }
    31 int d[2][1111][1<<10],VAL[1<<10];
    32 int main(){
    33     idx['A']=0; idx['G']=1; idx['T']=2; idx['C']=3;
    34     int m,n,a;
    35     char str[111];
    36     int val[11];
    37     while(~scanf("%d%d",&m,&n)){
    38         tn=0;
    39         memset(ch,0,sizeof(ch));
    40         memset(flag,0,sizeof(flag));
    41         for(int i=0; i<m; ++i){
    42             scanf("%s%d",str,val+i);
    43             if(strlen(str)>100) continue;
    44             insert(str,i);
    45         }
    46         for(int i=1; i<(1<<m); ++i){
    47             for(int j=0; j<m; ++j){
    48                 if((i>>j)&1) VAL[i]=VAL[i^(1<<j)]+val[j];
    49             }
    50         }
    51         init();
    52         for(int j=0; j<=tn; ++j){
    53             for(int k=0; k<(1<<m); ++k) d[0][j][k]=-INF;
    54         }
    55         d[0][0][0]=0;
    56         for(int i=0; i<n; ++i){
    57             int x=i&1;
    58             for(int j=0; j<=tn; ++j){
    59                 for(int k=0; k<(1<<m); ++k) d[x^1][j][k]=-INF;
    60             }
    61             for(int j=0; j<=tn; ++j){
    62                 for(int k=0; k<(1<<m); ++k){
    63                     if(d[x][j][k]==-INF) continue;
    64                     for(int y=0; y<4; ++y){
    65                         d[x^1][ch[j][y]][k|flag[ch[j][y]]]=max(d[x^1][ch[j][y]][k|flag[ch[j][y]]],d[x][j][k]+VAL[k^(k|flag[ch[j][y]])]);
    66                     }
    67                 }
    68             }
    69         }
    70         int res=-INF;
    71         for(int i=0; i<=tn; ++i){
    72             for(int j=0; j<(1<<m); ++j) res=max(res,d[n&1][i][j]);
    73         }
    74         if(res<0) puts("No Rabbit after 2012!");
    75         else printf("%d
    ",res);
    76     }
    77     return 0;
    78 }
  • 相关阅读:
    .net core2.0 中使用aspectcore实现aop
    [Superset] 设置Superset的登录设置
    [Python]Pandas对于非唯一的label index的数据选择问题
    Data Science Radar测试结果
    [R]R包版本更迭【持续更新】
    [面试] 删除多余的数组内容
    [Python]Python中的包(Package)
    [Linux] 使用Yum在CentOS上安装MySQL
    [pyMongo]insert_many的Bulkwrite实现机制
    [Git]2018-10 解决git cmd中文乱码问题
  • 原文地址:https://www.cnblogs.com/WABoss/p/5171530.html
Copyright © 2011-2022 走看看