zoukankan      html  css  js  c++  java
  • bzoj 4044 Virus synthesis

    题目传送门

      需要高级权限的传送门

    题目大意

      要求用两种操作拼出一个长度为$n$的只包含'A','T','G','C'的字符串

    1. 在当前字符串头或字符串结尾添加一个字符
    2. 将当前字符串复制,将复制的串翻转,接在当前字符串末尾或者当前字符串头部、

      问最少的操作次数。

      考虑正常的动态规划时怎么做的。用$f[i][j]$表示拼出给定串的$[l, r]$的最少用的步数。转移显然。

      然而它的时间爆了。空间也爆了。

      仔细观察发现,翻转完最后一个偶回文串后,两端的字符只能暴力插入。

      但是一个串的本质不同的回文串的总数是$O(n)$的级别。因此可以考虑在回文树上动态规划。

      然后用$f[x] + n - len(x)$去更新答案。

      转移显然要分奇偶性讨论。

    1. 如果当前的回文串是奇回文串,那么通过它的最长回文后缀转移,然后暴力补上缺字符,即$f[x] = f[fail[x]] + len(x) - len(fail[x])$。
    2. 如果当前的回文串是偶回文串,那么继续讨论
      • 可以通过它的长度不超过它的一半的最长回文后缀补上一半的残缺字符,然后翻转转移。
      • 如果通过它去掉两端字符的偶回文串在翻转之前再补上一个字符转移。即$f[y] + 1$

      另外说一句,奇回文串的$f[x]$设为$len(x)$也没有任何影响。

      求$half$的做法可以参考NOI某年的动物园。

    Code

      1 /**
      2  * bzoj
      3  * Problem#4044
      4  * Accepted
      5  * Time: 3264ms
      6  * Memory: 4512k
      7  */
      8 #include <bits/stdc++.h>
      9 using namespace std;
     10 typedef bool boolean;
     11 
     12 int cti(char x) {
     13     if (x == 'A')    return 0;
     14     if (x == 'G')    return 1;
     15     if (x == 'T')    return 2;
     16     return 3;
     17 }
     18 
     19 const int N = 1e5 + 5;
     20 
     21 typedef class TrieNode {
     22     public:
     23         int len, f;
     24         TrieNode* ch[4];
     25         TrieNode *fail, *half;
     26 }TrieNode;
     27 
     28 TrieNode pool[N];
     29 TrieNode* top;
     30 
     31 TrieNode* newnode(int len) {
     32     top->len = len;
     33     memset(top->ch, 0, sizeof(top->ch));
     34     return top++;
     35 }
     36 
     37 typedef class PalindromeTree {
     38     public:
     39         char *str;
     40         TrieNode *even, *odd;
     41         TrieNode *last;
     42         
     43         PalindromeTree() {    }
     44         PalindromeTree(char *str):str(str) {
     45             top = pool;
     46             odd = newnode(-1);
     47             even = newnode(0);
     48             odd->fail = odd, even->fail = odd, last = odd;
     49             odd->half = odd, even->half = odd;
     50         }
     51 
     52         TrieNode* extend(TrieNode* p, int pos) {
     53             while (str[pos] != str[pos - p->len - 1]) p = p->fail;
     54             return p;
     55         }
     56 
     57         void build() {
     58             for (int i = 1; str[i]; i++) {
     59                 int c = cti(str[i]);
     60                 last = extend(last, i);
     61                 if (!last->ch[c]) {
     62                     TrieNode* pn = newnode(last->len + 2);
     63                     pn->fail = extend(last->fail, i)->ch[c];
     64                     if (!pn->fail)
     65                         pn->fail = even;
     66                     TrieNode* f = last->half;
     67                     f = extend(f, i)->ch[c];
     68                     if (!f)
     69                         f = even;
     70                     while (f->len * 2 > pn->len)
     71                         f = f->fail;
     72                     pn->half = f;
     73                     last->ch[c] = pn;
     74                 }
     75                 last = last->ch[c];
     76             }
     77         }
     78 
     79         void dp() {
     80             for (TrieNode* p = pool; p < top; p++)
     81                 p->f = p->len;
     82             for (TrieNode* p = pool + 2; p < top; p++)
     83                 if (p->len & 1)
     84                     p->f = p->fail->f + (p->len - p->fail->len);
     85                 else {
     86                     p->f = min(p->f, p->half->f + ((p->len >> 1) - p->half->len) + 1);
     87                     for (int c = 0; c < 4; c++)
     88                         if (p->ch[c])
     89                             p->ch[c]->f = min(p->ch[c]->f, p->f + 1);
     90                 }
     91         }
     92 }PalindromeTree;
     93 
     94 int T;
     95 char str[N];
     96 PalindromeTree pam;
     97 
     98 inline void init() {
     99     scanf("%s", str + 1);
    100 }
    101 
    102 inline void solve() {
    103     pam = PalindromeTree(str);
    104     pam.build();
    105     pam.dp();
    106     int res = 142857, len = strlen(str + 1);
    107 //    for (TrieNode* p = pool; p < top; p++)
    108 //        cerr << p - pool << " " << p->fail - pool << " " << p->half - pool << " " << p->len << endl;
    109     for (TrieNode* p = pool; p < top; p++)
    110         res = min(res, len - p->len + p->f);
    111     printf("%d
    ", res);
    112 }
    113 
    114 int main() {
    115     scanf("%d", &T);
    116     while (T--) {
    117         init();
    118         solve();
    119     }
    120     return 0;
    121 }
  • 相关阅读:
    服务方式加载卸载NT驱动函数集
    《Windows核心编程》学习笔记(12)– 虚拟内存
    《Windows核心编程》学习笔记(14)– 堆
    数据库连接错误:提示TCP端口1433,sql server 2008 Connection refused:connect
    Windows驱动开发技术详解笔记
    Struts2文件上传的大小限制问题
    pragma comment的使用 pragma预处理指令详解
    解决FastCGI Error Error Number: 2147467259 (0x80004005). 和 Error Number: 1073741819 (0xc0000005).
    PHP中的日期处理
    mysql远程连接10061错误
  • 原文地址:https://www.cnblogs.com/yyf0309/p/8678599.html
Copyright © 2011-2022 走看看