[整理]Landau 定理的证明和应用

0. 定理内容

Landau 定理:竞赛图强连通的充要条件是按入度不降排序后除总和外的所有前缀 \(i\) 之和不等于 \(\dbinom{i}{2}\)

1. 证明

引理:强连通竞赛图的充要条件是对于任何一个真子集 \(S\),一定有 \(S\) 外的点到 \(S\) 有边。
必要性是显然的,接下来我们看一下充分性。我们可以先找到一个真子集 \(S\),然后对于集合外面的每个点找到一条连到集合的路径,此时考虑 \(S\) 的补集,则又可以从 \(S\) 出发连到某个点成环,这样一定可以扩展到整个图强连通。
有了这个引理,我们就知道图不连通的充要条件是有一个集合没有入度,也就是集合内点的入度之和等于边数。而假设有一些这样的集合,入度最小的一个前缀一定是其中之一,所以可以排序后直接判断。

2. 应用

CF1268D

这道题事实上需要更多性质,Landau 定理只是一小部分。
引理:一个至少 \(4\) 个点的强连通竞赛图存在点数少 \(1\) 的强连通子图。
证明可以考虑往 \(n-1\) 个点的竞赛图上添加一个点 \(n\),对于竞赛图,我们通常考虑缩点的处理手法,所以先把 \(n-1\) 个点的图缩成 DAG。

  1. 如果只有一个点那么原图直接满足条件。
  2. 如果缩点后的点数 \(tot>2\),那么一定有一条 \(tot\to n\to 1\) 的路径,可以在 \((1,\ tot)\) 中随便删一个。
  3. 如果缩点后的点数 \(tot=2\),那么至少有一个强连通分量内部有多于一个点,我们找到这个强连通分量中直接连 \(n\) 的点,假设在 \(1\) 里,那么从这个点出发连向每个 \(1\) 中的点会形成一棵叶向树,我们随便删一个叶子显然并不会影响剩下图的连通性。如果在 \(2\) 里改成根向树即可,引理得证。

由这个引理可以得出至少 \(4\) 个点的强连通竞赛图可以翻转一个点使得仍然强连通,只需要翻转上面证明中要删除的点即可。
接下来我们就离正解很近了,只需要考虑任意竞赛图上会发生什么即可,还是先缩点。
按照上面一样的套路讨论:

  1. 如果只有一个强连通分量不用翻。
  2. 如果有至少三个强连通分量就从中间找一个点翻转,此时一定成环。
  3. 如果只有两个强连通分量,就必须有一个连通分量有至少 \(4\) 个点,此时随便翻转一个可以保证这个分量仍然强连通且整个图成环强连通。

由于第三种情况需要有一个强连通分量至少 \(4\) 个点,所以总结起来就是至少 \(7\) 个点的竞赛图可以翻转至多一次,这样 \(n\ge7\) 时枚举翻哪个,\(n<7\) 时枚举翻转的集合。
Landau 定理终于在这里派上用场了:如果 Tarjan 判强连通会因为边数 \(O(n^2)\) 爆炸,而运用 Landau 定理可以减少到 \(O(n\log n)\)\(O(n)\)(基数排序),这样 \(n\ge7\) 时是 \(O(n^2\log n)\)\(n<7\) 时我写得非常丑是 \(O(2^nn^2)\),均可通过。
代码:

const int N=2010, p=998244353;
int n; char a[N][N];
int d[N], fac[N], td[N];
il bool Check(int S){
  if(S<0){
    rep(i, 1, n)d[i]=td[i];
    rep(i, 1, n)if(a[i][-S]==49)d[i]++, d[-S]--;
    rep(i, 1, n)if(a[-S][i]==49)d[i]--, d[-S]++;
  }else if(S>0){
    memset(d, 0, sizeof d);
    rep(i, 1, n){
      if((S>>i-1)&1)rep(j, 1, n)swap(a[i][j], a[j][i]);
    }
    rep(i, 1, n)rep(j, 1, n)if(a[i][j]==49)d[j]++;
  }else if(S==0)rep(i, 1, n)d[i]=td[i];
  sort(d+1, d+1+n);
  bool ff=1; int sum=0;
  rep(i, 1, n-1){
    sum+=d[i];
    if(sum==i*(i-1)/2)ff=0;
  }
  if(S>0){
    rep(i, 1, n){
      if((S>>i-1)&1)rep(j, 1, n)swap(a[i][j], a[j][i]);
    }
  }
  return ff;
}
signed main(){
  Read(n), fac[0]=1;
  rep(i, 1, n)scanf("%s", a[i]+1), fac[i]=fac[i-1]*i%p;
  rep(i, 1, n)rep(j, 1, n)if(a[i][j]==49)td[j]++;
  int ans=0;
  if(n<7){
    int mn=INF;
    rep(S, 0, (1ll<<n)-1){
      if(Check(S))mn=min(mn, (int)__builtin_popcountll(S));
    }
    if(mn==INF)return puts("-1"), 0;
    rep(S, 0, (1ll<<n)-1){
      if(__builtin_popcountll(S)==mn&&Check(S))(ans+=fac[mn])%=p;
    }
    printf("%lld %lld\n", mn, ans);
  }else {
    if(Check(0))return puts("0 1"), 0;
    rep(i, 1, n)if(Check(-i))ans++;
    if(!ans)puts("-1");
    else printf("1 %lld\n", ans);
  }
  return 0;
}
posted @ 2022-09-01 16:05  ajthreac  阅读(196)  评论(0编辑  收藏  举报