首先肯定先枚举数字
然后考虑二分答案
每个字符串向它合法的位置连边
然后易发现每个点出度最多为 n n n,不然没意义
所以最多 O ( n 2 ) O(n^2) O(n2) 条边
然后跑网络流,看能不能流完,也就是能不能匹配成功即可
#define M 200010
//#define mo
#define N 510
int n, m, i, j, k, T, ans=2e9;
char s[M];
int le[N][11], l, r, mid, c, tot;
vector<int>v[N][11];
map<int, int>mp;
int check(int T) {
mf_graph<int>G(N*N); mp.clear(); tot=n;
for(i=1; i<=n; ++i) G.add_edge(N*N-2, i, 1);
for(i=1; i<=n; ++i) {
for(j=0, k=0; j<=n; ++j) {
k=v[i][c][j%le[i][c]]+m*(j/le[i][c]);
if(k>T) break;
if(!mp[k+n])
mp[k+n]=++tot, G.add_edge(tot, N*N-1, 1);
G.add_edge(i, mp[k+n], 1);
}
}
if(G.flow(N*N-2, N*N-1, n)==n) return 1;
return 0;
}
signed main()
{
// freopen("in.txt", "r", stdin);
// freopen("out.txt", "w", stdout);
// srand(time(NULL));
// T=read();
// while(T--) {
//
// }
n=read(); m=read();
for(i=1; i<=n; ++i) {
scanf("%s", s);
for(j=0; s[j]; ++j) v[i][s[j]-'0'].pb(j);
}
for(c=0; c<=9; ++c) for(i=1; i<=n; ++i)
le[i][c]=v[i][c].size();
for(c=0; c<=9; ++c) {
for(i=1; i<=n; ++i) if(!le[i][c]) break;
if(i<=n) continue;
l=n-1; r=1e9;
while(l<r) {
mid=(l+r)>>1;
if(check(mid)) r=mid;
else l=mid+1;
}
ans=min(ans, l);
}
if(ans==2e9) printf("-1");
else printf("%d", ans);
return 0;
}