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\n",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('\n');
      return 0;
  }
      
View Code

 

posted @ 2015-08-21 14:02  maijing  阅读(282)  评论(0编辑  收藏  举报