zoukankan      html  css  js  c++  java
  • bzoj 1195

    http://www.lydsy.com/JudgeOnline/problem.php?id=1195

    状压DP。

    首先去掉被包含的字符串。

    对于字符串i和j,我们求出 当字符串j的左端点在字符串i的左端点的左边或与字符串i的左端点重合时,字符串i和字符串j可以重合的最长长度cost是多少。

    就是求下面红色部分的最长长度cost:

    这个强行枚举就可以了,反正数据这么小。

    注意,因为我们已经去掉了被包含的字符串,所以不会出现下面这种情况:

    所以去掉了被包含的字符串是为了保证当左端点单调时,右端点也是单调的

    建一个图,我们在图中i连到j一条费用为cost的有向边。

    然后就是求不重复经过点,可以走的最长路径。

    这是哈密顿路径问题,为NP问题,但是这道题数据范围很小,可以用状压DP。

    对于输出字典序最小字符串那里,我们在找决策的时候比较一下即可。

    #include<cstdio>
    #include<cstdlib>
    #include<iostream>
    #include<fstream>
    #include<algorithm>
    #include<cstring>
    #include<string>
    #include<cmath>
    #include<queue>
    #include<stack>
    #include<map>
    #include<utility>
    #include<set>
    #include<bitset>
    #include<vector>
    #include<functional>
    #include<deque>
    #include<cctype>
    #include<climits>
    #include<complex>
    //#include<bits/stdc++.h>适用于CF,UOJ,但不适用于poj
     
    using namespace std;
    
    typedef long long LL;
    typedef double DB;
    typedef pair<int,int> PII;
    typedef complex<DB> CP;
    
    #define mmst(a,v) memset(a,v,sizeof(a))
    #define mmcy(a,b) memcpy(a,b,sizeof(a))
    #define fill(a,l,r,v) fill(a+l,a+r+1,v)
    #define re(i,a,b)  for(i=(a);i<=(b);i++)
    #define red(i,a,b) for(i=(a);i>=(b);i--)
    #define ire(i,x) for(typedef(x.begin()) i=x.begin();i!=x.end();i++)
    #define fi first
    #define se second
    #define m_p(a,b) make_pair(a,b)
    #define SF scanf
    #define PF printf
    
    
    template<class T>inline T sqr(T x){return x*x;}
    template<class T>inline void upmin(T &t,T tmp){if(t>tmp)t=tmp;}
    template<class T>inline void upmax(T &t,T tmp){if(t<tmp)t=tmp;}
    
    const DB EPS=1e-9;
    inline int sgn(DB x){if(abs(x)<EPS)return 0;return(x>0)?1:-1;}
    const DB Pi=acos(-1.0);
    
    inline int gint()
      {
            int res=0;bool neg=0;char z;
            for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
            if(z==EOF)return 0;
            if(z=='-'){neg=1;z=getchar();}
            for(;z!=EOF && isdigit(z);res=res*10+z-'0',z=getchar());
            return (neg)?-res:res; 
        }
    inline LL gll()
      {
          LL res=0;bool neg=0;char z;
            for(z=getchar();z!=EOF && z!='-' && !isdigit(z);z=getchar());
            if(z==EOF)return 0;
            if(z=='-'){neg=1;z=getchar();}
            for(;z!=EOF && isdigit(z);res=res*10+z-'0',z=getchar());
            return (neg)?-res:res; 
        }
    
    const int maxN=12;
    const int maxlen=50;
    
    int N;
    char s[maxN+10][maxlen+10];
    
    int tempN;
    char temps[maxN+10][maxlen+10];
    int f[maxN+10];
    
    inline int smaller(char *s1,char *s2,int l1,int l2)
      {
          int i,len1=strlen(s1+1),len2=strlen(s2+1);
          re(i,1,min(len1-l1+1,len2-l2+1))if(s1[l1+i-1]!=s2[l2+i-1])return s1[l1+i-1]<s2[l2+i-1];
          return len1-l1+1<len2-l2+1;
      }
    
    inline int same(char *s1,char *s2,int l1,int l2,int len)
      {
          int i;
          if(l1+len-1>strlen(s1+1))return 0;
          if(l2+len-1>strlen(s2+1))return 0;
          re(i,1,len)if(s1[l1+i-1]!=s2[l2+i-1])return 0;
          return 1;
      }
    
    inline int check(char *s1,char *s2)
      {
          int i,l1=strlen(s1+1),l2=strlen(s2+1);
          re(i,1,l2-l1+1)if(same(s1,s2,1,i,l1))return 1;
          return 0;
      }
    
    int now,first[maxN+10];
    struct Tedge{int v,cost,next;}edge[maxN*maxN+100];
    inline void addedge(int u,int v,int cost){now++;edge[now].v=v;edge[now].cost=cost;edge[now].next=first[u];first[u]=now;}
    
    #define two(k) (1<<((k)-1))
    #define wei(v,k) ((v>>(k-1))&1)
    
    int F[maxN+10][(1<<maxN)+100],vis[maxN+10][(1<<maxN)+100];
    int head,tail;PII que[maxN*(1<<maxN)+100000];
    
    int cnt;char out[maxN*maxlen+1000];
    
    int main()
      {
          /*freopen("substr.in","r",stdin);
          freopen("substr.out","w",stdout);*/
          int i,j;
          N=gint();
          re(i,1,N)SF("%s
    ",s[i]+1);
          re(i,1,N)re(j,1,N)if(i!=j)if(check(s[i],s[j])){f[i]=1;break;}
          mmcy(temps,s);
            tempN=N;N=0;
          re(i,1,tempN)if(!f[i])mmcy(s[++N],temps[i]);
          if(N==0)N=1;
          
          now=-1;mmst(first,-1);
          re(i,1,N)re(j,1,N)if(i!=j)
            {
                int leni=strlen(s[i]+1),lenj=strlen(s[j]+1),res=lenj;
                while(res!=0 && !same(s[i],s[j],1,lenj-res+1,res))res--;
                addedge(i,j,res);
            }
          
          mmst(F,-1);mmst(vis,0);
          head=0;tail=-1;
          re(i,1,N)F[i][two(i)]=0,vis[i][two(i)]=1,que[++tail]=PII(i,two(i));
          while(head<=tail)
            {
                int u=que[head%(maxN*(1<<maxN)+100000)].fi,state=que[head%(maxN*(1<<maxN)+100000)].se,v,cost;head++;
                vis[u][state]=0;
                for(i=first[u],v=edge[i].v,cost=edge[i].cost;i!=-1;i=edge[i].next,v=edge[i].v,cost=edge[i].cost)
                  if(!wei(state,v) && F[u][state]+cost>F[v][state+two(v)])
                    {
                        F[v][state+two(v)]=F[u][state]+cost;
                        if(!vis[v][state+two(v)])
                          {
                              vis[v][state+two(v)]=1;
                              que[(++tail)%(maxN*(1<<maxN)+100000)]=PII(v,state+two(v));
                          }
                    }
            }
          
          now=-1;mmst(first,-1);
          re(i,1,N)re(j,1,N)if(i!=j)
            {
                int leni=strlen(s[i]+1),lenj=strlen(s[j]+1),res=lenj;
                while(res!=0 && !same(s[i],s[j],1,lenj-res+1,res))res--;
                addedge(j,i,res);
            }
          
          int u=-1,state=two(N+1)-1;
          re(i,1,N)if(u==-1 || F[u][state]<F[i][state] || (F[u][state]==F[i][state] && smaller(s[i],s[u],1,1)))u=i;
          re(i,1,strlen(s[u]+1))out[++cnt]=s[u][i];
          for(int T=N-1;T;T--)
            {
                int p=-1,o,v,cost;
                for(i=first[u],v=edge[i].v,cost=edge[i].cost;i!=-1;i=edge[i].next,v=edge[i].v,cost=edge[i].cost)
                  if(wei(state,v) && F[v][state-two(u)]+cost==F[u][state])
                    if(p==-1 || smaller(s[v],s[p],cost+1,o))
                      p=v,o=cost+1;
                re(i,o,strlen(s[p]+1))out[++cnt]=s[p][i];
                    state-=two(u);
                u=p;
            }
          re(i,1,cnt)putchar(out[i]);putchar('
    ');
          return 0;
      }
          
    View Code
  • 相关阅读:
    ORACLE DROP TABLE和truncate table的区别
    C#版链表加强版
    C#版栈
    再谈为什么要使用MONO
    流浪猫伏击大白鹅
    编写ASP.NET复合控件实例
    C# 大文件拷贝
    关于团队项目构架设计的疑问
    在Windows平台下使用MONO
    C#版链表
  • 原文地址:https://www.cnblogs.com/maijing/p/4747719.html
Copyright © 2011-2022 走看看