若僅保留這$k$個點仍然有環,那麼顯然無解。blog
不然設$A$表示這$k$個點的集合,$B$表示剩下的點的集合,由於是競賽圖,每一個集合內部的拓撲關係是一條鏈,方便起見將全部點按照在所在集合的鏈上的位置進行重標號。ip
對於$B$中每一個點$i$,求出$l_i$表示最小的$j$,知足$B_i\rightarrow A_j$有邊,再求出$r_i$表示最大的$j$,知足$A_j\rightarrow B_i$有邊。io
那麼須要保留$B$中儘量多的點,知足對於任意兩個點$B_j,B_i(j\leq i)$都有$r_j<l_i$,否則就會有環,即$l_i>\max(r_j)(j\leq i)$。class
考慮DP,設$f[i][j]$表示考慮$B$中前$i$個點,前面選擇的點的$r$的最大值爲$j$時最多能夠選擇幾個點,暴力轉移便可。im
時間複雜度$O(n^2)$。sort
#include<cstdio> const int N=2010,BUF=12000000; char Buf[BUF],*buf=Buf; int n,m,i,j,x,y,l[N],r[N],L,R,a[N],ca,b[N],cb,q[N],h,t,d[N],cnt;bool g[N][N],vip[N]; int f[N][N],ans; inline void read(int&a){for(a=0;*buf<48;buf++);while(*buf>47)a=a*10+*buf++-48;} void toposort(int S){ for(i=1;i<=n;i++)d[i]=0; for(i=1;i<=n;i++)if(vip[i]==S)for(j=1;j<=n;j++)if(vip[j]==S)d[j]+=g[i][j]; for(h=i=1,t=cnt=0;i<=n;i++)if(vip[i]==S){ cnt++; if(!d[i])q[++t]=i; } while(h<=t)for(x=q[h++],i=1;i<=n;i++)if(vip[i]==S&&g[x][i])if(!(--d[i]))q[++t]=i; } inline void up(int&a,int b){a<b?(a=b):0;} inline int max(int a,int b){return a>b?a:b;} int main(){ fread(Buf,1,BUF,stdin);read(n),read(m); for(i=1;i<=n;i++)for(j=1;j<=n;j++)read(x),g[i][j]=x; for(i=1;i<=m;i++)read(x),vip[x]=1; toposort(1); if(t<cnt)return puts("impossible"),0; for(i=1;i<=cnt;i++)a[++ca]=q[i]; toposort(0); for(i=1;i<=cnt;i++)b[++cb]=q[i]; for(i=1;i<=ca;i++)d[a[i]]=i; for(i=1;i<=cb;i++){ x=b[i]; L=ca+1,R=0; for(j=1;j<=n;j++)if(vip[j]){ y=d[j]; if(g[x][j]){ if(y<L)L=y; }else if(y>R)R=y; } for(j=0;j<=ca;j++)f[i][j]=f[i-1][j]; if(L>R)for(j=0;j<L;j++)up(f[i][j>R?j:R],f[i-1][j]+1); } for(j=0;j<=ca;j++)up(ans,f[cb][j]); ans=cb-ans; if(ans>=m)puts("impossible");else printf("%d",ans); return 0; }